Exchanger是Java中用于两个线程交换数据的工具,通过exchange()方法在汇合点互换对象,支持超时设置,适用于双线程协同场景。

在Java并发编程中,Exchanger 是一个用于两个线程之间交换数据的同步工具类。它位于 java.util.concurrent 包下,允许两个线程在某个汇合点交换各自持有的对象,非常适合双向数据传递的场景。
Exchanger 的基本原理
Exchangerexchange(V data) 方法,当第一个线程调用时会等待,直到第二个线程也调用该方法,此时两个线程交换数据并继续执行。
这种机制类似于“碰头交易”:两个线程各带一份数据,在约定地点交换后离开。
简单使用示例
下面是一个使用 Exchanger 在两个线程间交换字符串的例子:
立即学习“Java免费学习笔记(深入)”;
import java.util.concurrent.Exchanger;
public class ExchangerExample {
public static void main(String[] args) {
Exchanger exchanger = new Exchanger<>();
new Thread(() -> {
try {
String data = "来自线程 A 的数据";
System.out.println("线程 A 准备交换数据");
String received = exchanger.exchange(data);
System.out.println("线程 A 接收到: " + received);
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
}
}, "Thread-A").start();
new Thread(() -> {
try {
String data = "来自线程 B 的数据";
System.out.println("线程 B 准备交换数据");
String received = exchanger.exchange(data);
System.out.println("线程 B 接收到: " + received);
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
}
}, "Thread-B").start();
}
}
输出结果可能是:
线程 A 准备交换数据 线程 B 准备交换数据 线程 A 接收到: 来自线程 B 的数据 线程 B 接收到: 来自线程 A 的数据
带超时的 exchange 方法
除了基本的 exchange(V data),Exchanger 还提供了一个带超时时间的方法:exchange(V data, long timeout, TimeUnit unit)。
如果在指定时间内没有另一个线程来交换数据,该方法将抛出 TimeoutException 或返回 null(取决于实现),避免无限等待。
例如:
try {
String received = exchanger.exchange(data, 5, TimeUnit.SECONDS);
} catch (InterruptedException | TimeoutException e) {
System.err.println("交换超时或被中断");
}
适用场景与注意事项
Exchanger 适用于以下场景:
- 两个线程需要周期性地交换缓冲区数据,比如生产者-消费者模式中的双缓冲技术
- 工作线程之间协同处理任务并交换中间结果
- 实现高性能的数据传输通道
使用时注意:
- 只能用于两个线程之间的交换,多于两个线程调用 exchange 可能导致不确定行为
- 线程必须成对出现,否则会阻塞等待
- 交换的数据可以为 null,但需确保逻辑上能正确处理
基本上就这些。Exchanger 虽然使用场景较特定,但在合适的场合下能简化线程通信逻辑,提升代码可读性和性能。不复杂但容易忽略。











