Java多线程之2-- CyclicBarrier的用法

本文通过一个实例详细介绍了CyclicBarrier的使用方法,演示了如何利用该工具类使多个线程同步到达某个公共障碍点,从而确保所有线程都能在特定时刻一起开始执行后续任务。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

CyclicBarrier是一个同步辅助类,它允许一组线程互相等待,直到到达某个公共屏障点 (common barrier point)

CyclicBarrier和CountDownLatch一样,都是关于线程的计数器。只不过CountDownLatch是控制在结束的时候,而CyclicBarrier是控制在线程开始的时候。

package thread;

import java.text.SimpleDateFormat;
import java.util.Arrays;
import java.util.Date;
import java.util.List;
import java.util.Random;
import java.util.concurrent.CyclicBarrier;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.TimeUnit;

public class CyclicBarrierDemo {
    final static SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");

    public static void main(String[] args) throws InterruptedException {
        Worker worker = null;
        ExecutorService excutor = null;
        try {
            List<String> names = Arrays.asList("张三", "李四", "王五", "牛六", "孙七");
            // 此处用来控制同组线程的cyclicBarrier,注意构造方法里面的参数必须和生成线程的总数相等,否则多余的线程就会一直处于等待执行状态
            CyclicBarrier cyclicBarrier = new CyclicBarrier(names.size());// 多人同时准备
            excutor = Executors.newCachedThreadPool();
            // 线程名
            String name = null;
            for (int i = 0; i < names.size(); i++) {
                name = names.get(i);
                // 线程初始化
                worker = new Worker(name, cyclicBarrier);
                excutor.execute(worker);
                TimeUnit.SECONDS.sleep(2);
            }
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            if (excutor != null) {
                excutor.shutdown();
            }
        }
    }

    static class Worker implements Runnable {
        String workerName;
        CyclicBarrier cyclicBarrier;
        Random random;

        public Worker(String workerName, CyclicBarrier cyclicBarrier) {
            this.workerName = workerName;
            this.cyclicBarrier = cyclicBarrier;
            this.random = new Random();
        }

        public void run() {
            try {
                int sleepTime = random.nextInt(10);
                TimeUnit.SECONDS.sleep(sleepTime);
                System.out.println(
                        "**** " + workerName + " after sleep " + sleepTime + " , ready at " + sdf.format(new Date()));
                // 线程在这里等待,直到所有线程都到达公共障碍点
                cyclicBarrier.await();
                System.out.println("---- " + workerName + "  await at " + sdf.format(new Date()));
                TimeUnit.SECONDS.sleep(sleepTime);
                System.out.println("++++ " + workerName + " after sleep " + sleepTime + " , sleep at " + sdf.format(new Date()));

            } catch (Exception e) {
                e.printStackTrace();
            }

        }
    }

}

执行结果如下图,可以通过执行结果看出5条线程的run的时间是不一样的,而且在run方法中,执行到cyclicBarrier.await();时候都停住了等待最后一条线程执行到此屏障点,即**** 孙七 after sleep 4 , ready at 2017-02-23 17:36:04 这个时刻点。然后继续执行剩下的代码,你会从结果中发现只有在await的时候的时间点都是一样的。具体看执行结果:

**** 张三 after sleep 3 , ready at 2017-02-23 17:35:55
**** 李四 after sleep 2 , ready at 2017-02-23 17:35:56
**** 王五 after sleep 6 , ready at 2017-02-23 17:36:02
**** 牛六 after sleep 6 , ready at 2017-02-23 17:36:04
**** 孙七 after sleep 4 , ready at 2017-02-23 17:36:04
---- 孙七  await at 2017-02-23 17:36:04
---- 张三  await at 2017-02-23 17:36:04
---- 李四  await at 2017-02-23 17:36:04
---- 王五  await at 2017-02-23 17:36:04
---- 牛六  await at 2017-02-23 17:36:04
++++ 李四 after sleep 2 , sleep at 2017-02-23 17:36:06
++++ 张三 after sleep 3 , sleep at 2017-02-23 17:36:07
++++ 孙七 after sleep 4 , sleep at 2017-02-23 17:36:08
++++ 王五 after sleep 6 , sleep at 2017-02-23 17:36:10
++++ 牛六 after sleep 6 , sleep at 2017-02-23 17:36:10



评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值