如何在Java中使用Exchanger实现线程间数据交换

Exchanger是Java中用于两个线程间双向交换数据的同步工具,通过exchange()方法在汇合点交换数据,一个线程调用后会阻塞等待另一个线程配对交换,成功后各自获得对方数据并继续执行,适用于双缓冲、数据校验等成对线程协作场景,不支持多于两个线程的直接使用,且需注意阻塞和中断处理。

在Java中,Exchanger 是一个用于两个线程之间双向交换数据的同步工具类,位于 java.util.concurrent 包下。它提供了一种简单而高效的方式,让两个线程在某个汇合点交换各自的数据,常用于生产者-消费者场景、数据校对、线程间协作等。

Exchanger的基本原理

Exchanger 是一个泛型类,其中 V 表示要交换的数据类型。两个线程通过调用 exchange(V data) 方法来传递自己的数据,当双方都调用了 exchange 方法时,数据就会自动交换,方法返回对方线程传来的数据。

如果一个线程先到达 exchange 点,它会阻塞等待另一个线程调用 exchange。一旦配对成功,两个线程立即获得对方的数据并继续执行。

使用Exchanger的步骤与示例

下面是一个简单的例子,展示两个线程如何使用 Exchanger 交换字符串数据:

import java.util.concurrent.Exchanger;

public class ExchangerExample {
    public static void main(String[] args) {
        Exchanger exchanger = new Exchanger<>();

        Thread thread1 = new Thread(() -> {
            try {
                String data1 = "来自线程1的数据";
                System.out.println("线程1准备交换: " + data1);
                
                String received = exchanger.exchange(data1);
                System.out.println("线程1收到: " + received);
            } catch (InterruptedException e) {
                Thread.currentThread().interrupt();
            }
        });

        Thread thread2 = new Thread(() -> {
            try {
                String data2 = "来自线程2的数据";
                System.out.println("线程2准备交换: " + data2);
                
                String received = exchanger.exchange(data2);
                System.out.println("线程2收到: " + received);
            } catch (InterruptedException e) {
                Thread.currentThread().interrupt();
            }
        });

        thread1.start();
        thread2.start();
    }
}
  

输出可能如下(顺序可能略有不同):

线程1准备交换: 来自线程1的数据
线程2准备交换: 来自线程2的数据
线程1收到: 来自线程2的数据
线程2收到: 来自线程1的数据

实际应用场景与注意事项

Exchanger 适用于需要成对线程协作的场景,比如:

  • 双缓冲数据交换:一个线程填充缓冲区,另一个处理,完成后交换缓冲区引用
  • 数据校验:两个线程分别计算结果,在某个时间点交换并比对
  • 游戏中的角色状态同步:两个玩家线程定期交换状态信息

注意点:

  • 只能用于两个线程之间的数据交换,多于两个线程无法直接使用
  • exchange 方法是阻塞的,直到配对线程也调用 exchange
  • 可设置超时版本:exchange(V data, long timeout, TimeUnit unit),避免无限等待
  • 若一个线程中断,另一个调用 exchange 的线程会抛出 InterruptedException

基本上就这些。Exchanger 使用简单,但在合适的场景下能极大简化线程间协作逻辑。关键是理解它的“成对交换”机制,避免误用于多个线程或单向通信场景。