线程间交换数据的工具:Exchanger

Exchanger 是一个用于线程间协作的工具类,用于两个线程间能够交换。它提供了一个交换的同步点,在这个同步点两个线程能够交换数据。具体交换数据是通过 exchange 方法来实现的,如果一个线程先执行 exchange 方法,那么它会同步等待另一个线程也执行 exchange 方法,这个时候两个线程就都达到了同步点,两个线程就可以交换数据。

Exchanger 除了一个无参的构造方法外,主要方法也很简单:

  1. //当一个线程执行该方法的时候,会等待另一个线程也执行该方法,因此两个线程就都达到了同步点
  2. //将数据交换给另一个线程,同时返回获取的数据
  3. V exchange(V x) throws InterruptedException
  4. //同上一个方法功能基本一样,只不过这个方法同步等待的时候,增加了超时时间
  5. V exchange(V x, long timeout, TimeUnit unit)
  6. throws InterruptedException, TimeoutException

一个例子

Exchanger 理解起来很容易,这里用一个简单的例子来看下它的具体使用。我们来模拟这样一个情景:在青春洋溢的中学时代,下课期间,男生经常会给走廊里为自己喜欢的女孩子送情书,相信大家都做过这样的事情吧 :)。男孩会先到女孩教室门口,然后等女孩出来,教室那里就是一个同步点,然后彼此交换信物,也就是彼此交换了数据。现在,就来模拟这个情景。

  1. public class ExchangerDemo {
  2. private static Exchanger<String> exchanger = new Exchanger();
  3. public static void main(String[] args) {
  4. //代表男生和女生
  5. ExecutorService service = Executors.newFixedThreadPool(2);
  6. service.execute(() -> {
  7. try {
  8. //男生对女生说的话
  9. String girl = exchanger.exchange("我其实暗恋你很久了......");
  10. System.out.println("女孩儿说:" + girl);
  11. } catch (InterruptedException e) {
  12. e.printStackTrace();
  13. }
  14. });
  15. service.execute(() -> {
  16. try {
  17. System.out.println("女生慢慢的从教室里走出来......");
  18. TimeUnit.SECONDS.sleep(3);
  19. //男生对女生说的话
  20. String boy = exchanger.exchange("我也很喜欢你......");
  21. System.out.println("男孩儿说:" + boy);
  22. } catch (InterruptedException e) {
  23. e.printStackTrace();
  24. }
  25. });
  26. }
  27. }

输出结果:

  1. 女生慢慢的从教室里走出来......
  2. 男孩儿说:我其实暗恋你很久了......
  3. 女孩儿说:我也很喜欢你......

这个例子很简单,也很能说明Exchanger的基本使用。当两个线程都到达调用exchange方法的同步点的时候,两个线程就能交换彼此的数据。