Exchanger是Java中用于两个线程间双向数据交换的工具,通过exchange()方法在汇合点交换对象,适用于双缓冲、生产者-消费者等场景,支持阻塞等待与超时机制,确保线程安全高效协作。

在Java并发编程中,Exchanger 是一个用于两个线程之间双向数据交换的同步工具类。它允许两个线程在某个汇合点交换各自持有的对象,常用于生产者-消费者场景或数据缓冲对的切换操作。
Exchanger 的基本原理
Exchanger 可以看作是一个“交换站”。两个线程分别调用 exchange(V) 方法时,会阻塞等待对方到达,一旦双方都调用了该方法,它们持有的数据就会被交换并返回对方的数据。
关键特性:
- 必须是两个线程参与交换,多于或少于两个都无法完成配对。
- 交换是双向的:每个线程都能拿到对方传来的数据。
- 如果一个线程先到达,它会阻塞等待另一个线程调用
exchange()。
如何使用 Exchanger 进行线程间数据交换
下面通过一个简单示例演示两个线程如何使用 Exchanger 交换字符串数据:
立即学习“Java免费学习笔记(深入)”;
import java.util.concurrent.Exchanger;public class ExchangerExample { public static void main(String[] args) { Exchanger
exchanger = new Exchanger<>(); Thread threadA = new Thread(() -> { String data = "来自线程A的数据"; try { System.out.println("线程A准备交换..."); String received = exchanger.exchange(data); System.out.println("线程A收到: " + received); } catch (InterruptedException e) { Thread.currentThread().interrupt(); } }); Thread threadB = new Thread(() -> { String data = "来自线程B的数据"; try { System.out.println("线程B准备交换..."); String received = exchanger.exchange(data); System.out.println("线程B收到: " + received); } catch (InterruptedException e) { Thread.currentThread().interrupt(); } }); threadA.start(); threadB.start(); }}
运行结果可能是:
线程A准备交换...
线程B准备交换...
线程A收到: 来自线程B的数据
线程B收到: 来自线程A的数据说明两个线程成功完成了数据互换。
实际应用场景:双缓冲数据交换
一个典型的应用是双缓冲机制,比如一个线程负责填充数据缓冲区,另一个线程处理已填满的缓冲区,然后交换空/满缓冲区指针。
import java.util.concurrent.Exchanger; import java.util.ArrayList;public class BufferExchange { static class DataBuffer { ArrayList
data = new ArrayList<>(); boolean inUse = false; void add(int value) { data.add(value); } void clear() { data.clear(); } } public static void main(String[] args) { Exchangerexchanger = new Exchanger<>(); DataBuffer buffer1 = new DataBuffer(); DataBuffer buffer2 = new DataBuffer(); Thread producer = new Thread(() -> { DataBuffer currentBuffer = buffer1; for (int i = 0; i < 5; i++) { currentBuffer.clear(); for (int j = 0; j < 3; j++) { currentBuffer.add((i * 3) + j); } System.out.println("生产者填充完毕: " + currentBuffer.data); try { // 与消费者交换缓冲区 currentBuffer = exchanger.exchange(currentBuffer); System.out.println("生产者获得新缓冲区,继续填充"); } catch (InterruptedException e) { Thread.currentThread().interrupt(); } } }); Thread consumer = new Thread(() -> { DataBuffer currentBuffer = buffer2; try { for (int i = 0; i < 5; i++) { // 等待生产者填充后交换 currentBuffer = exchanger.exchange(currentBuffer); System.out.println("消费者处理数据: " + currentBuffer.data); // 处理完成后交还空缓冲区 } } catch (InterruptedException e) { Thread.currentThread().interrupt(); } }); producer.start(); consumer.start(); } }
在这个例子中,生产者和消费者通过
Exchanger高效地交换缓冲区,避免了频繁创建对象和锁竞争。注意事项与使用建议
-
只能两个线程配对:若多个线程调用同一个
Exchanger,系统会随机选择一对进行交换,其余继续等待,可能导致不可预期的行为。 -
支持超时交换:
exchange(V, timeout, unit)可设置最大等待时间,防止无限阻塞。 - 传递 null 值合法:可以交换 null,但需注意空指针处理。
- 适合周期性数据同步场景,如数据采集与分析、双缓冲刷新等。
基本上就这些。Exchanger 虽然不如 CountDownLatch 或 CyclicBarrier 常见,但在特定协作模式下非常简洁高效。不复杂但容易忽略。










