JDK中为了处理线程之间的同步问题,除了提供锁机制以外,还提供了几个很是有用的并发工具类:CountDownLatch、CyclicBarrier、Semphore、Exchanger、Phaser;
CountDownLatch、CyclicBarrier、Semphore、Phaser 这四个工具类提供一种并发流程的控制手段;而Exchanger工具类则提供了在线程之间交换数据的一种手段。html
Exchanger的功能是使2个线程之间交换数据(有很多文章的说法是“传输数据”,应该叫“交换数据”更合适,由于这是两个线程都要向对方传送数据,同时也获取对方的传送过来的数据,是java
public V exchange(V x) throws InterruptedException
等待另外一个线程到达此交换点(除非当前线程被中断),而后将给定的对象传送给该线程,并接收该线程的对象。
public V exchange(V x, long timeout, TimeUnit unit) throws InterruptedException, TimeoutException
等待另外一个线程到达此交换点(除非当前线程被中断,或者超出了指定的等待时间),而后将给定的对象传送给该线程,同时接收该线程的对
算法
如下是重点介绍的一个类,该类使用 Exchanger 在线程间交换缓冲区,所以,在须要时,填充缓冲区的线程获取一个新腾空的缓冲区,并将填满的缓冲区传递给腾空缓冲区的线程bash
class FillAndEmpty {
Exchanger<DataBuffer> exchanger = new Exchanger<DataBuffer>();
DataBuffer initialEmptyBuffer = ... a made-up type
DataBuffer initialFullBuffer = ...
class FillingLoop implements Runnable {
public void run() {
DataBuffer currentBuffer = initialEmptyBuffer;
try {
while (currentBuffer != null) {
addToBuffer(currentBuffer);
if (currentBuffer.isFull())
currentBuffer = exchanger.exchange(currentBuffer);
}
} catch (InterruptedException ex) { ... handle ... }
}
}
class EmptyingLoop implements Runnable {
public void run() {
DataBuffer currentBuffer = initialFullBuffer;
try {
while (currentBuffer != null) {
takeFromBuffer(currentBuffer);
if (currentBuffer.isEmpty())
currentBuffer = exchanger.exchange(currentBuffer);
}
} catch (InterruptedException ex) { ... handle ...}
}
}
void start() {
new Thread(new FillingLoop()).start();
new Thread(new EmptyingLoop()).start();
}
}
复制代码
Exchanger能够用于并发
Exchanger也能够用于ide
好比咱们须要将纸制银流经过人工的方式录入成电子银行流水,为了不错误,采用AB岗两人进行录入,录入到Excel以后,系统须要加载这两个Excel,并对这两个Excel数据进行校对,看看是否录入的一致。代码以下:工具
private static final Exchanger<String> exgr = new Exchanger<String>();
private static ExecutorService threadPool = Executors.newFixedThreadPool(2);
public static void main(String[] args) {
threadPool.execute(new Runnable() {
@Override
public void run() {
try {
String A = "银行流水A";// A录入银行流水数据
exgr.exchange(A);//同步点,交换数据
} catch (InterruptedException e) {
}
}
});
threadPool.execute(new Runnable() {
@Override
public void run() {
try {
String B = "银行流水B";// B录入银行流水数据
String A = exgr.exchange("B");//同步点,交换数据
System.out.println("A和B数据是否一致:" + A.equals(B) + "\nA录入的是:"+ A + "\nB录入的是:" + B);
} catch (InterruptedException e) {
}
}
});
threadPool.shutdown();
}
复制代码
运行结果:
oop
A和B数据是否一致:false
A录入的是:银行流水A
B录入的是:银行流水B
复制代码
文章源地址:https://www.cnblogs.com/jinggod/p/8494384.htmlspa