【并发技术16】线程同步工具Exchanger的使用

简介: 【并发技术16】线程同步工具Exchanger的使用

如果两个线程在运行过程中需要交换彼此的信息,比如一个数据或者使用的空间,就需要用到 Exchanger 这个类,Exchanger 为线程交换信息提供了非常方便的途径,它可以作为两个线程交换对象的同步点,只有当每个线程都在进入 exchange() 方法并给出对象时,才能接受其他线程返回时给出的对象。

每次只能两个线程交换数据,如果有多个线程,也只有两个能交换数据。下面看个通俗的例子:一手交钱一首交货!

public class ExchangerTest { 
    public static void main(String[] args) { 
        ExecutorService service = Executors.newCachedThreadPool(); 
        final Exchanger exchanger = new Exchanger(); //定义一个交换对象,用来交换数据       
        //开启一个线程执行任务
        service.execute(new Runnable(){             
            @Override
            public void run() { 
                try {                
                    String data1 = "海洛因"; 
                    System.out.println("线程" + Thread.currentThread().getName() 
                            + "正在把毒品" + data1 + "拿出来");                    
                    Thread.sleep((long)(Math.random()*10000)); 
                  //把要交换的数据传到exchange方法中,然后被阻塞,等待另一个线程与之交换。返回交换后的数据
                    String data2 = (String)exchanger.exchange(data1); 
                    System.out.println("线程" + Thread.currentThread().getName() +  
                    "用海洛因换来了" + data2); 
                }catch(Exception e){     
                } finally {
                    service.shutdown();
                    System.out.println("交易完毕,拿着钱快跑!");
                }
            }    
        }); 
        //开启另一个线程执行任务
        service.execute(new Runnable(){             
            @Override
            public void run() { 
                try {                
                    String data1 = "300万"; 
                    System.out.println("线程" + Thread.currentThread().getName() +  
                    "正在把" + data1 +"拿出来"); 
                    Thread.sleep((long)(Math.random()*10000));                       
                    String data2 = (String)exchanger.exchange(data1);                     
                    System.out.println("线程" + Thread.currentThread().getName() +  
                    "用300万弄到了" + data2); 
                }catch(Exception e){     
                } finally {
                    service.shutdown();
                    System.out.println("交易完毕,拿着海洛因快跑!");
                }
            }    
        });         
    } 
}

从代码中我仿佛看到了两个人在交易毒品的场面……来看一下交易结果如何:

线程pool-1-thread-1正在把毒品海洛因拿出来

线程pool-1-thread-2正在把300万拿出来

线程pool-1-thread-2用300万弄到了海洛因

线程pool-1-thread-1用海洛因换来了300万

交易完毕,拿着海洛因快跑!

交易完毕,拿着钱快跑!

跑的倒是快,从运行结果来看,确实实现了数据的交换,这个只是交换一个基本类型的数据而已,它真正的用处不仅仅局限于此,比如我们可以交换一个对象,这就有用了,JDK 官方提到了一个高级的应用:

使用 Exchanger 在线程间交换缓冲区,因此,在需要时,填充缓冲区的线程获取一个新腾空的缓冲区,并将填满的缓冲区传递给腾空缓冲区的线程。

这就得根据实际情况了,思想和上面的一样,实际中肯定要定义一个缓冲区的类,然后两个线程之间交换的就是这个缓冲区的类即可,至于类中如何实现,就得看实际情况了。Exchanger 的使用就总结这么多吧~

如果两个线程在运行过程中需要交换彼此的信息,比如一个数据或者使用的空间,就需要用到 Exchanger 这个类,Exchanger 为线程交换信息提供了非常方便的途径,它可以作为两个线程交换对象的同步点,只有当每个线程都在进入 exchange() 方法并给出对象时,才能接受其他线程返回时给出的对象。

每次只能两个线程交换数据,如果有多个线程,也只有两个能交换数据。下面看个通俗的例子:一手交钱一首交货!

public class ExchangerTest { 
    public static void main(String[] args) { 
        ExecutorService service = Executors.newCachedThreadPool(); 
        final Exchanger exchanger = new Exchanger(); //定义一个交换对象,用来交换数据       
        //开启一个线程执行任务
        service.execute(new Runnable(){             
            @Override
            public void run() { 
                try {                
                    String data1 = "海洛因"; 
                    System.out.println("线程" + Thread.currentThread().getName() 
                            + "正在把毒品" + data1 + "拿出来");                    
                    Thread.sleep((long)(Math.random()*10000)); 
                  //把要交换的数据传到exchange方法中,然后被阻塞,等待另一个线程与之交换。返回交换后的数据
                    String data2 = (String)exchanger.exchange(data1); 
                    System.out.println("线程" + Thread.currentThread().getName() +  
                    "用海洛因换来了" + data2); 
                }catch(Exception e){     
                } finally {
                    service.shutdown();
                    System.out.println("交易完毕,拿着钱快跑!");
                }
            }    
        }); 
        //开启另一个线程执行任务
        service.execute(new Runnable(){             
            @Override
            public void run() { 
                try {                
                    String data1 = "300万"; 
                    System.out.println("线程" + Thread.currentThread().getName() +  
                    "正在把" + data1 +"拿出来"); 
                    Thread.sleep((long)(Math.random()*10000));                       
                    String data2 = (String)exchanger.exchange(data1);                     
                    System.out.println("线程" + Thread.currentThread().getName() +  
                    "用300万弄到了" + data2); 
                }catch(Exception e){     
                } finally {
                    service.shutdown();
                    System.out.println("交易完毕,拿着海洛因快跑!");
                }
            }    
        });         
    } 
}

从代码中我仿佛看到了两个人在交易毒品的场面……来看一下交易结果如何:

线程pool-1-thread-1正在把毒品海洛因拿出来

线程pool-1-thread-2正在把300万拿出来

线程pool-1-thread-2用300万弄到了海洛因

线程pool-1-thread-1用海洛因换来了300万

交易完毕,拿着海洛因快跑!

交易完毕,拿着钱快跑!

跑的倒是快,从运行结果来看,确实实现了数据的交换,这个只是交换一个基本类型的数据而已,它真正的用处不仅仅局限于此,比如我们可以交换一个对象,这就有用了,JDK 官方提到了一个高级的应用:

使用 Exchanger 在线程间交换缓冲区,因此,在需要时,填充缓冲区的线程获取一个新腾空的缓冲区,并将填满的缓冲区传递给腾空缓冲区的线程。

这就得根据实际情况了,思想和上面的一样,实际中肯定要定义一个缓冲区的类,然后两个线程之间交换的就是这个缓冲区的类即可,至于类中如何实现,就得看实际情况了。Exchanger 的使用就总结这么多吧~


相关文章
|
2月前
|
安全
List并发线程安全问题
【10月更文挑战第21天】`List` 并发线程安全问题是多线程编程中一个非常重要的问题,需要我们认真对待和处理。只有通过不断地学习和实践,我们才能更好地掌握多线程编程的技巧和方法,提高程序的性能和稳定性。
249 59
|
2月前
|
安全 Java
线程安全的艺术:确保并发程序的正确性
在多线程环境中,确保线程安全是编程中的一个核心挑战。线程安全问题可能导致数据不一致、程序崩溃甚至安全漏洞。本文将分享如何确保线程安全,探讨不同的技术策略和最佳实践。
58 6
|
2月前
|
安全 Java 开发者
Java 多线程并发控制:深入理解与实战应用
《Java多线程并发控制:深入理解与实战应用》一书详细解析了Java多线程编程的核心概念、并发控制技术及其实战技巧,适合Java开发者深入学习和实践参考。
85 7
|
2月前
|
存储 安全 Java
Java多线程编程中的并发容器:深入解析与实战应用####
在本文中,我们将探讨Java多线程编程中的一个核心话题——并发容器。不同于传统单一线程环境下的数据结构,并发容器专为多线程场景设计,确保数据访问的线程安全性和高效性。我们将从基础概念出发,逐步深入到`java.util.concurrent`包下的核心并发容器实现,如`ConcurrentHashMap`、`CopyOnWriteArrayList`以及`BlockingQueue`等,通过实例代码演示其使用方法,并分析它们背后的设计原理与适用场景。无论你是Java并发编程的初学者还是希望深化理解的开发者,本文都将为你提供有价值的见解与实践指导。 --- ####
|
2月前
|
安全 程序员 API
|
2月前
|
存储 设计模式 分布式计算
Java中的多线程编程:并发与并行的深度解析####
在当今软件开发领域,多线程编程已成为提升应用性能、响应速度及资源利用率的关键手段之一。本文将深入探讨Java平台上的多线程机制,从基础概念到高级应用,全面解析并发与并行编程的核心理念、实现方式及其在实际项目中的应用策略。不同于常规摘要的简洁概述,本文旨在通过详尽的技术剖析,为读者构建一个系统化的多线程知识框架,辅以生动实例,让抽象概念具体化,复杂问题简单化。 ####
|
2月前
|
存储 安全 UED
多线程在打包工具中的运用
【11月更文挑战第2天】本文介绍了多线程技术在打包工具中的应用,包括提高打包效率、优化用户体验和多线程安全考虑。通过并行处理文件和加速资源收集,多线程可以显著缩短打包时间。在用户体验方面,多线程使界面保持响应,并支持优先级处理。此外,文章还讨论了资源访问冲突和死锁预防的解决方案,确保多线程环境下的稳定性和安全性。
|
3月前
|
Java
【编程进阶知识】揭秘Java多线程:并发与顺序编程的奥秘
本文介绍了Java多线程编程的基础,通过对比顺序执行和并发执行的方式,展示了如何使用`run`方法和`start`方法来控制线程的执行模式。文章通过具体示例详细解析了两者的异同及应用场景,帮助读者更好地理解和运用多线程技术。
45 1
|
28天前
|
NoSQL Redis
单线程传奇Redis,为何引入多线程?
Redis 4.0 引入多线程支持,主要用于后台对象删除、处理阻塞命令和网络 I/O 等操作,以提高并发性和性能。尽管如此,Redis 仍保留单线程执行模型处理客户端请求,确保高效性和简单性。多线程仅用于优化后台任务,如异步删除过期对象和分担读写操作,从而提升整体性能。
62 1
|
3月前
|
存储 消息中间件 资源调度
C++ 多线程之初识多线程
这篇文章介绍了C++多线程的基本概念,包括进程和线程的定义、并发的实现方式,以及如何在C++中创建和管理线程,包括使用`std::thread`库、线程的join和detach方法,并通过示例代码展示了如何创建和使用多线程。
72 1

相关实验场景

更多