CountDownLatch(门闩)和CyclicBarrier(栅栏)
CountDownLatch 实现线程等待
如何实现主线程需要等待子线程运行完毕才能运行,Java 并发包里已经提供了实现类似功能的工具类:CountDownLatch?
定义:让一些线程阻塞直到另一些线程完成一些列操作之后才被唤醒。即使门闩的意思,给定一个数,当这个数被减到0时才能开始运行另一个线程。
主要有两个方法一个是await方法等待,一个是countDown方法让门闩减一。
可以简单理解为:
从自助餐拿的吃的必须吃完才会开门放人离开,不然打断腿。
public class Test06_CountDownLatchDemo {
static CountDownLatch countDownLatch = new CountDownLatch(5);
public static void main(String[] args) throws InterruptedException {
for(int i = 0;i<5;i++){
final int temp = i;
new Thread(()->{
System.out.println(Thread.currentThread().getName()+'\t'+"吃完了蛋糕"+temp);
countDownLatch.countDown();
}).start();
}
countDownLatch.await();
System.out.println("吃完了");
}
}
CyclicBarrier 实现线程同步
CountDownLatch 主要用来解决一个线程等待多个线程的场景,可以类比旅游团团长要等待所有的游客到齐才能去下一个景点;CyclicBarrier 是一组线程之间互相等待,更像是几个驴友之间不离不弃。除此之外 CountDownLatch 的计数器是不能循环利用的,也就是说一旦计数器减到 0,再有线程调用 await(),该线程会直接通过。但 CyclicBarrier 的计数器是可以循环利用的,而且具备自动重置的功能,一旦计数器减到 0 会自动重置到你设置的初始值。除此之外,CyclicBarrier 还可以设置回调函数。
需要对应个数的线程都完成相应的操作后,才能顺序执行。
public class T07_TestCyclicBarrier {
public static void main(String[] args){
CyclicBarrier barrie = new CyclicBarrier(20,()->{
System.out.println("满人");
});
for(int i = 0;i<100;i++){
new Thread(()->{
try {
barrie.await();
} catch (InterruptedException e) {
e.printStackTrace();
} catch (BrokenBarrierException e) {
e.printStackTrace();
}
}).start();
}
}
}