举例说明CyclicBarrier循环屏障:
游乐园水上划船,规定5个人划一艘船,如果人不够,那么就需要等待,直到凑齐5个人了,才允许开始。也就是说,5个人一波5个人一波。每一波都是同一时间开始玩的。
即:循环屏障保证了多个线程在准确的同一刻开始。
注:CyclicBarrier循环屏障适用于多线程必须同时开始的情景。
代码示例
import java.util.Date;
import java.util.concurrent.BrokenBarrierException;
import java.util.concurrent.CyclicBarrier;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
/**
* 循环屏障 示例
*
* @author JustryDeng
* @date 2018/10/17 19:39
*/
public class CyclicBarrierDemo {
/** 设置循环屏障数 */
private static CyclicBarrier cyclicBarrier = new CyclicBarrier(5);
public static void main(String[] args) throws InterruptedException {
// -> 创建可缓存长线程池
ExecutorService executorService = Executors.newCachedThreadPool();
for (int i = 1; i <= 10; i++) {
Thread.sleep(1000);
// 使用lambel表达式简单实现Runnable接口的run方法
executorService.execute(() -> {
try {
System.out.println(new Date() + "线程" + Thread.currentThread().getName() + "准备就绪!");
// 设置屏障点
cyclicBarrier.await();
System.out.println(new Date() + "线程" + Thread.currentThread().getName() + "开始运行!");
} catch (InterruptedException e) {
e.printStackTrace();
} catch (BrokenBarrierException e) {
e.printStackTrace();
}
});
}
// 关闭线程池
executorService.shutdown();
}
}
运行主函数,输出结果为:
注:await(long timeout, TimeUnit unit)方法可以设置一个超时时间,如果等待超时了,那么会抛出异常;其
它同组等待的线程就知道不需要等了,没有结果的。
注:CyclicBarrier常用场景有(举例说明):
1.双十一00:00分秒杀
2.准点抢票
3.跑分测试
……
注:本人多线程一栏的文章大多为简记、后续时间充裕了再回头进行丰富。
注:循环屏障与线程池搭配使用时,为了避免程序死锁(假设说明:线程池最多同时运行的线程数为100, 你循环屏障要求
的线程数是101,那么线程池里面的100个线程任务就会等着,等着够101个线程后才开始后面的逻辑,由于线程池
的线程数上限就是100,也就是那100个线程永远等不到第101个线程了,永远不会满足屏障数了,进而造成死锁),
一般需要保证 线程池核心线程数 >= 屏障数。
追注:如果线程池最大线程数大于线程池核心线程数,且线程池等待队列容量为0,此时只需要保
证线程池最大线程数 >= 屏障数。
笔者将本人多线程一栏中博客涉及到的所有代码示例(Lock分开放在一个专门的项目、synchronized的代码附在该文章末尾),放在GIT上了(链接见本文末),这里先给出一个所涉及内容图:
声明:本文是学习笔记,主要学习自以下视频。