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

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

  1. public class ExchangerTest {

  2.    public static void main(String[] args) {

  3.        ExecutorService service = Executors.newCachedThreadPool();

  4.        final Exchanger exchanger = new Exchanger(); //定义一个交换对象,用来交换数据      

  5.        //开启一个线程执行任务

  6.        service.execute(new Runnable(){            

  7.            @Override

  8.            public void run() {

  9.                try {                

  10.                    String data1 = "海洛因";

  11.                    System.out.println("线程" + Thread.currentThread().getName()

  12.                            + "正在把毒品" + data1 + "拿出来");                    

  13.                    Thread.sleep((long)(Math.random()*10000));

  14.                  //把要交换的数据传到exchange方法中,然后被阻塞,等待另一个线程与之交换。返回交换后的数据

  15.                    String data2 = (String)exchanger.exchange(data1);

  16.                    System.out.println("线程" + Thread.currentThread().getName() +  

  17.                    "用海洛因换来了" + data2);

  18.                }catch(Exception e){    

  19.                } finally {

  20.                    service.shutdown();

  21.                    System.out.println("交易完毕,拿着钱快跑!");

  22.                }

  23.            }    

  24.        });

  25.        //开启另一个线程执行任务

  26.        service.execute(new Runnable(){            

  27.            @Override

  28.            public void run() {

  29.                try {                

  30.                    String data1 = "300万";

  31.                    System.out.println("线程" + Thread.currentThread().getName() +  

  32.                    "正在把" + data1 +"拿出来");

  33.                    Thread.sleep((long)(Math.random()*10000));                      

  34.                    String data2 = (String)exchanger.exchange(data1);                    

  35.                    System.out.println("线程" + Thread.currentThread().getName() +  

  36.                    "用300万弄到了" + data2);

  37.                }catch(Exception e){    

  38.                } finally {

  39.                    service.shutdown();

  40.                    System.out.println("交易完毕,拿着海洛因快跑!");

  41.                }

  42.            }    

  43.        });        

  44.    }

  45. }

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

线程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() 方法并给出对象时,才能接受其他线程返回时给出的对象。

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

  1. public class ExchangerTest {

  2.    public static void main(String[] args) {

  3.        ExecutorService service = Executors.newCachedThreadPool();

  4.        final Exchanger exchanger = new Exchanger(); //定义一个交换对象,用来交换数据      

  5.        //开启一个线程执行任务

  6.        service.execute(new Runnable(){            

  7.            @Override

  8.            public void run() {

  9.                try {                

  10.                    String data1 = "海洛因";

  11.                    System.out.println("线程" + Thread.currentThread().getName()

  12.                            + "正在把毒品" + data1 + "拿出来");                    

  13.                    Thread.sleep((long)(Math.random()*10000));

  14.                  //把要交换的数据传到exchange方法中,然后被阻塞,等待另一个线程与之交换。返回交换后的数据

  15.                    String data2 = (String)exchanger.exchange(data1);

  16.                    System.out.println("线程" + Thread.currentThread().getName() +  

  17.                    "用海洛因换来了" + data2);

  18.                }catch(Exception e){    

  19.                } finally {

  20.                    service.shutdown();

  21.                    System.out.println("交易完毕,拿着钱快跑!");

  22.                }

  23.            }    

  24.        });

  25.        //开启另一个线程执行任务

  26.        service.execute(new Runnable(){            

  27.            @Override

  28.            public void run() {

  29.                try {                

  30.                    String data1 = "300万";

  31.                    System.out.println("线程" + Thread.currentThread().getName() +  

  32.                    "正在把" + data1 +"拿出来");

  33.                    Thread.sleep((long)(Math.random()*10000));                      

  34.                    String data2 = (String)exchanger.exchange(data1);                    

  35.                    System.out.println("线程" + Thread.currentThread().getName() +  

  36.                    "用300万弄到了" + data2);

  37.                }catch(Exception e){    

  38.                } finally {

  39.                    service.shutdown();

  40.                    System.out.println("交易完毕,拿着海洛因快跑!");

  41.                }

  42.            }    

  43.        });        

  44.    }

  45. }

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

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

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

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

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

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

交易完毕,拿着钱快跑!

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

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

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

作者:华为云云享专家倪升武

【并发技术16】线程同步工具Exchanger的使用的更多相关文章

  1. 线程同步工具 Semaphore类使用案例

    参考博文 : 线程同步工具(一) 线程同步工具(二)控制并发访问多个资源 并发工具类(三)控制并发线程数的Semaphore 使用Semaphore模拟互斥锁 当一个线程想要访问某个共享资源,首先,它 ...

  2. 线程同步工具 Semaphore类的基础使用

    推荐好文: 线程同步工具(一) 线程同步工具(二)控制并发访问多个资源 并发工具类(三)控制并发线程数的Semaphore 简介 Semaphore是基于计数的信号量,可以用来控制同时访问特定资源的线 ...

  3. Java核心知识点学习----线程同步工具类,CyclicBarrier学习

    线程同步工具类,CyclicBarrier日常开发较少涉及,这里只举一个例子,以做备注.N个人一块出去玩,相约去两个地方,CyclicBarrier的主要作用是等待所有人都汇合了,才往下一站出发. 1 ...

  4. Python并发编程-进程 线程 同步锁 线程死锁和递归锁

    进程是最小的资源单位,线程是最小的执行单位 一.进程 进程:就是一个程序在一个数据集上的一次动态执行过程. 进程由三部分组成: 1.程序:我们编写的程序用来描述进程要完成哪些功能以及如何完成 2.数据 ...

  5. 同步工具——Exchanger

    本博客系列是学习并发编程过程中的记录总结.由于文章比较多,写的时间也比较散,所以我整理了个目录贴(传送门),方便查阅. 并发编程系列博客传送门 本文是转载文章,原文请见这里 一.Exchanger简介 ...

  6. 并发编程之 线程协作工具 LockSupport

    前言 在前面的文章中,我们介绍了并发工具中的4个,Samephore,CyclicBarrier,CountDownLatch,Exchanger,但是我们漏了一个,非常的好用的工具,楼主在这里必须加 ...

  7. Java提高班(三)并发中的线程同步与锁

    乐观锁.悲观锁.公平锁.自旋锁.偏向锁.轻量级锁.重量级锁.锁膨胀...难理解?不存的!来,话不多说,带你飙车. 上一篇介绍了线程池的使用,在享受线程池带给我们的性能优势之外,似乎也带来了另一个问题: ...

  8. java并发编程:线程同步和锁

    一.锁的原理 java中每个对象都有一个内置锁.当程序运行到非静态的synchronized同步方法上时,自动获得与正在执行代码类的当前实例(this)有关的锁.获得一个对象的锁也称为获取锁,当程序运 ...

  9. C++并发编成 03 线程同步

    这一节主要讲讲线程同步的方式,C++ 11中提供了丰富的线程同步元语,如condition_variable,futrue,std::packaged_task<>,std::promis ...

随机推荐

  1. CSPS模拟 51

    蒟蒻由于仍然苟活在$1jf$,不得不接受省选题的吊打$QWQ$ 蒟蒻由于拿了大神们不屑打的弱智暴力,而大神们$T3$的各种快速变换没调出来,所以拿到辽人生第一个$1jf$黄名 既侥幸又$kx$ T1 ...

  2. NOIP模拟 6

    考试时看了看T1,觉得是结论题,推了推没有成果,跑去看第二题, 题意很明确,求过定点的最小环,还没思考解题策略,然后觉得是水题 打了个tarjan找边双(据说会炸但是平均表现良好),在边双里暴力拆边找 ...

  3. EffectiveJava-1

    最近在看EffectiveJava,记录一下,分享一下自己的心得. 一.将局部变量的作用于最小化 在第一次使用的地方进行声明,过早的声明局部变量,会延长局部变量的生命周期,若在代码块外声明变量,当程序 ...

  4. 【IOS开发—视图】

    一.UIWindow对象 每一个app都有一个UIWindow对象,它像一个容器一样,用来包含应用中的所有视图,应用会在启动时创建并设置UIWindow对象. - (BOOL)application: ...

  5. 使用Typescript重构axios(二十九)——添加baseURL

    0. 系列文章 1.使用Typescript重构axios(一)--写在最前面 2.使用Typescript重构axios(二)--项目起手,跑通流程 3.使用Typescript重构axios(三) ...

  6. 爬虫学习--Urllib库基本使用 Day1

    一.Urllib库详解 1.什么是Urllib Python内置的HTTP请求库 urllib.request    请求模块(模拟实现传入网址访问) urllib.error             ...

  7. linux 自启动 | 三种方式自启动

    linux 实现自启动有多种方式,通过Linux 底层启动原理介绍,便可以理解以下几种方式 这里简单介绍一下这几种方式 一.自定义开机程序   /etc/rc.d/rc.local  1.vim  / ...

  8. node.js 需要注意知识点

    复习node.js 需要注意知识点--(重点) 2.1:参数传递获取造型 客户端脚手架(发)    (参数传递)    node.js(收) -发ajax  this.axios.get(" ...

  9. 访问控制列表ACL

    1.ACL Access list ,访问控制列表. 2.作用 限制网络中的地址访问. 3.主要内容 Eg: Router(config)#access-list ? <一>. <1 ...

  10. [笔记] HOW2J.CN网站记录的java笔记_第四部分_HTML

    1.写一个显示Hello World的网页, <html> <body> <p>Hello World</p> </body> </h ...