如果两个线程在运行过程中需要交换彼此的信息,比如一个数据或者使用的空间,就需要用到 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. 接口自动化、移动端、web端自动化如何做?

    1.<Python+Appium移动端自动化项目实战>-带您进入APP自动化测试的世界https://yuedu.baidu.com/ebook/765b38a5690203d8ce2f0 ...

  2. PHP yaf显示错误提示

    PHP yaf显示错误提示 1就是配置文件的那个错误 <pre>error_reporting(E_ALL);</pre> 2init.php文件的<pre>fun ...

  3. K8S入门系列之集群yum安装(一)

    kubernetes master 节点包含的组件: 1.kube-apiserver :集群核心,集群API接口.集群各个组件通信的中枢:集群安全控制: 2.kube-scheduler: 集群调度 ...

  4. Java nio 空轮询bug到底是什么

    编者注:Java nio 空轮询bug也就是Java nio在Linux系统下的epoll空轮询问题. epoll机制是Linux下一种高效的IO复用方式,相较于select和poll机制来说.其高效 ...

  5. PHP读取Excel内的图片

    今天接到了一个从Excel内读取图片的需求,在网上查找了一些资料,基本实现了自己的需求,不过由于查到的一些代码比较久远,不能直接移植到自己的项目里,需要稍加改动一下. 这里介绍一下分别使用phpspr ...

  6. 简单的倒叙应用---倒序打印字符串(C语言)

    void reverseStr(char* str){ if(*str=='\0'){ return; } reverseStr(str+1); printf("%c\n",*st ...

  7. 一个简单的C#爬虫程序

    这篇这篇文章主要是展示了一个C#语言如何抓取网站中的图片.实现原理就是基于http请求.C#给我们提供了HttpWebRequest和WebClient两个对象,方便发送请求获取数据,下面看如何实 1 ...

  8. VS 使用 :新建项目

    1.文件位置不放C盘

  9. windows版的mysql主从复制环境搭建

    背景 最近在学习用Spring Aop来实现数据库读写分离的功能. 在编写代码之前,首先是要部署好mysql的环境,因为要实现读写分离,所以至少需要部署两个mysql实例,一主一从,并且主从实例之间能 ...

  10. windows 10 上使用pybind11进行C++和Python代码相互调用 | Interfacing C++ and Python with pybind11 on windows 10

    本文首发于个人博客https://kezunlin.me/post/8b9c051d/,欢迎阅读! Interfacing C++ and Python with pybind11 on window ...