/**
* Exchanger让两个线程互换信息
* 实例模拟服务生和顾客,服务生往空杯子中倒水,顾客从装满水的杯子中喝水,然后互换杯子,服务生接着倒水,顾客接着喝水.
*/
/**
* 使用Exchanger的关键技术点如下:
* 1.初始化Exchanger对象时,可以通过泛型指定杯子能交换的信息类型.如"new Exchanger<String>;"表示只能交换String类型的信息
* 2.Exchanger的exchange方法表示当前线程准备交换信息,等待其他线程与它交换信息.当有其他线程调用该Exchanger对象的exchange方法时,立即交换信息
*/
public class ExchangerTest {
//描述一个装水的杯子
public static class Cup{
private boolean full = false; //标识杯子是否有水
public Cup(boolean full){
this.full = full;
}
//添水,假设需要5s
public void addWater(){
if(!this.full){
try{
Thread.sleep(5000);
}catch(InterruptedException e){
}
this.full = true;
}
}
//喝水,假设需要10s
public void drinkWater(){
if(this.full){
try{
Thread.sleep(10000);
}catch(InterruptedException e){
}
this.full = false;
}
}
}
public static void testExchanger(){
//初始化一个Exchanger,并规定可交换的信息类型是杯子
final Exchanger<Cup> exchanger = new Exchanger<Cup>();
//初始化一个空的杯子和装满水的杯子
final Cup initialEmptyCup = new Cup(false);
final Cup initialFullCup = new Cup(true);
//服务生线程
class Waiter implements Runnable{
public void run(){
Cup currentCup = initialEmptyCup;
try{
int i = 0;
while(i < 2){
System.out.println("服务生开始往杯子里倒水: " + System.currentTimeMillis());
//往空的杯子里倒水
currentCup.addWater();
System.out.println("服务生添水完毕: " + System.currentTimeMillis());
//杯子满后和顾客的空杯子交换
System.out.println("服务生等待与顾客交换杯子: " + System.currentTimeMillis());
currentCup = exchanger.exchange(currentCup);
System.out.println("服务生与顾客交换杯子完毕: " + System.currentTimeMillis());
i++;
}
}catch(InterruptedException ex){
}
}
}
//顾客线程
class Customer implements Runnable{
public void run(){
Cup currentCup = initialFullCup;
try{
int i = 0;
while(i < 2){
System.out.println("顾客开始喝水: " + System.currentTimeMillis());
//把杯子里的水喝掉
currentCup.drinkWater();
System.out.println("顾客喝水完毕: " + System.currentTimeMillis());
//将空杯子和服务生的满杯子交换
System.out.println("顾客等待与服务生交换杯子: " + System.currentTimeMillis());
exchanger.exchange(currentCup);
System.out.println("顾客与服务生交换杯子完毕: " + System.currentTimeMillis());
i++;
}
}catch(InterruptedException ex){
}
}
}
new Thread(new Waiter()).start();
new Thread(new Customer()).start();
}
public static void main(String... args){
ExchangerTest.testExchanger();
}
}
/**
Waiter是模拟服务生的线程,首先往空杯子中添水,然后调用Exchanger的exchange方法,等待和别人交换杯子.Customer是模拟了顾客的线程,首先把装满水的杯子喝光,然后调用Exchange的exchange方法,等待和别人交换杯子.当服务生和顾客都准备交换杯子时,由Exchanger将服务生手中装满水的杯子和顾客手中的空杯子交换.服务生可以继续倒水,而顾客可以继续喝水.
*/