CountDownLatch的简单使用

CountDownLatch 是一个同步工具类,它允许一个或多个线程一直等待(CountDownLatch.await()),直到其他线程执行完后再执行。使用一个计数器进行实现,计数器初始值为线程的数量。

当每一个线程完成自己任务后,计数器的值就会减一(CountDownLatch.countDown())。当计数器的值为0时,表示所有的线程都已经完成一些任务,然后在CountDownLatch上等待的线程就可以恢复执行接下来的任务。

public class TestThread {
    public static void main(String[] args) {
        ExecutorService executorService = Executors.newCachedThreadPool(); //线程池对象
        CountDownLatch countDownLatch=new CountDownLatch(10); //初始值为10
        for (int i = 0; i <10 ; i++) {
            executorService.execute(new countRunnable(countDownLatch));
        }
    }
}
class countRunnable implements Runnable{

    private CountDownLatch countDownLatch;

    public countRunnable(CountDownLatch countDownLatch) {
        this.countDownLatch = countDownLatch;
    }

    @Override
    public void run() {
        synchronized (countDownLatch){
            System.out.println("Thread Counts:"+countDownLatch.getCount()); //当前数值
            countDownLatch.countDown(); //数值-1
        }
        try {
            System.out.println("concurrency counts:"+(10-countDownLatch.getCount()));
            countDownLatch.await(); //阻塞当前线程
            System.out.println("线程执行完毕:"+countDownLatch.getCount());
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

CountDownLatch是线程安全的,底层是基于CAS机制(乐观锁),保证对单个变量的读-改-写是原子操作。

这里去掉synchronized (countDownLatch) 会出现重复重复打印的数字是因为countDownLatch.countDown()和countDownLatch.getCount()两个原子操作组合在一起就不是原子操作了。

 

public class EmployeeThread extends Thread{

    private CountDownLatch countDownLatch;
    private String name; //员工名字
    private Long time; //耗费时间

    public EmployeeThread(CountDownLatch countDownLatch, String name, Long time) {
        this.countDownLatch = countDownLatch;
        this.name = name;
        this.time = time;
    }

    @Override
    public void run() {
        try {
            System.out.println(name+"开始准备");
            Thread.sleep(time);
            System.out.println(name+"准备完成");
            countDownLatch.countDown();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}
public class TestEmployeeThread {
    public static void main(String[] args) {
        CountDownLatch countDownLatch=new CountDownLatch(2);
        EmployeeThread A=new EmployeeThread(countDownLatch,"A",1500L);
        EmployeeThread B=new EmployeeThread(countDownLatch,"B",1500L);
        EmployeeThread C=new EmployeeThread(countDownLatch,"C",1500L);

        B.start();
        C.start();
        try {
            countDownLatch.await(); //阻塞主线程

            //当B和C都执行完毕,countDownLatch减到0,主线程再开始执行

            System.out.println("B、C准备完成");  
            A.start();

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

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值