CountDownLatchCyclicBarrier同为Java1.5开始引入的,应用于多线程编程中的一种工具,二者用途十分相近,十分容易混淆。

CountDownLatch

CountDownLatch用来使一个线程等待其他N个线程执行完毕之后,在执行。在其他N个线程没有执行结束(调用countDown())之前一直阻塞。其他的N个线程各执行完毕之后各自退出。

例如,老王开了一家夜市店,不断有客人来吃饭,他必须等所有客人吃完饭之后他才能打烊。第一个客人走了,边直接走了,跟第二个客人走不走无关。第二个客人吃完也就自己走了,只有老板一个人在等着所有客人吃完离开,如果过了很晚之后客人还没走(超时了),则强制打烊,代码示例如下:

import java.util.concurrent.CountDownLatch;
import java.util.concurrent.TimeUnit; public class CountDownLatchTest { public static void main(String[] args) throws InterruptedException { final CountDownLatch latch=new CountDownLatch(2); Thread t1=new Thread(){
public void run(){
try {
System.out.println("第1桌客人进来吃饭...");
Thread.sleep(2000);//模拟任务耗时
System.out.println("第1桌客人进来吃完了...");
latch.countDown();
} catch (InterruptedException e) {
}
}
}; Thread t2=new Thread(){
public void run(){
try {
System.out.println("第2桌客人进来吃饭...");
Thread.sleep(3000);//模拟任务耗时
System.out.println("第2桌客人进来吃完了...");
latch.countDown();
} catch (InterruptedException e) {
}
}
};
Thread t3=new Thread(){
public void run(){
try {
System.out.println("我是店主:饭店正在营业,等客人吃完饭之后再打烊...");
latch.await(15, TimeUnit.SECONDS);
System.out.println("我是店主:你们已经吃完了,现在该我来收尾工作,然后打烊了...");
} catch (InterruptedException e) {
}
}
};
t3.start();
t1.start();
t2.start();
} }
  • 运行结果

    第1桌客人进来吃饭...

    我是店主:饭店正在营业,等客人吃完饭之后再打烊...

    第2桌客人进来吃饭...

    第1桌客人进来吃完了...

    第2桌客人进来吃完了...

    我是店主:你们已经吃完了,现在该我来收尾工作,然后打烊了...

这里启用了3个现场模拟,其中t3是那个等待线程,t3调用latch.await(15, TimeUnit.SECONDS); 之后便开始阻塞,等着t1,t2给信号,根据构造方法的定义CountDownLatch(2) 这里他需要等待的计数为2,t1,t2在执行完毕之后分别调用一次countDown(),而后t3便可以开始执行。假设这里t2调用sleep的时间为20秒,那么t3在等不到t2的countDown() 信号的情况下,会因为到达设置的15秒超时而强制打烊。

CyclicBarrier

CyclicBarrier的作用就是使多个线程达到同一个屏障然后才能继续后续工作。在这个过程中,各个线程是互相等待。大家都到达同一个屏障之后才能继续下一步的工作。

例如,朋友一起聚餐,大家提议要干杯。有人动作快,有人动作慢,动作快的人酒杯已经举起来了,然后就这么举着,等大家陆陆续续的都把杯子酒倒满,陆陆续续的举起来,最终当大家都举起来了之后,一起大喊:干!然后就各自喝了起来。这里的屏障就是大家都把杯子举起来的那一刻,那么在这之前,所有人都是阻塞的状态都在互相做自己的事情,只有当所有人都完成举杯,到达了这个一致的屏障,大家便可以喊:干杯了。代码示例如下:

import java.util.concurrent.BrokenBarrierException;
import java.util.concurrent.CyclicBarrier; public class CycleBarrierTest { public static void main(String[] args) { final CyclicBarrier cb=new CyclicBarrier(3); Thread t1=new Thread(){
public void run(){
try {
Thread.sleep(2000);
System.out.println("张三说:我的杯子已经端起来了");
cb.await();
System.out.println("张三说:干!");
} catch (InterruptedException e) {
e.printStackTrace();
} catch (BrokenBarrierException e) {
e.printStackTrace();
}
}
}; Thread t2=new Thread(){
public void run(){
try {
Thread.sleep(3000);
System.out.println("李四说:我的杯子已经端起来了");
cb.await();
System.out.println("李四说:干!");
} catch (InterruptedException e) {
e.printStackTrace();
} catch (BrokenBarrierException e) {
e.printStackTrace();
}
}
}; Thread t3=new Thread(){
public void run(){
try {
Thread.sleep(1000);
System.out.println("王五说:我的杯子已经端起来了");
cb.await();
System.out.println("王五说:干!");
} catch (InterruptedException e) {
e.printStackTrace();
} catch (BrokenBarrierException e) {
e.printStackTrace();
}
}
}; t1.start();
t2.start();
t3.start(); } }
  • 运行结果

    王五说:我的杯子已经端起来了

    张三说:我的杯子已经端起来了

    李四说:我的杯子已经端起来了

    李四说:干!

    王五说:干!

    张三说:干!

从运行的结果顺序看出,王五先把杯子举起来了并没有向下执行,打印“干”。而是大家都举起来之后,一起说出了“干”。这边是CyclicBarrier的效果。

这里从够赞函数CyclicBarrier(3)看出初始化计数为3,需要等待三个计数,只有有一个计数没达到那么会一直阻塞下去,如果这里把3改成4,会发现程序没法执行结束而是一直阻塞。

总结

上面两个示例主要展示了,CyclicBarrier和CountDownLatch的用法以及,CyclicBarrier场景下所有线程都在等待彼此达到共同屏障,而CountDownLatch中是一个线程阻塞着等待其他线程的信号。

二者还有一个最大的区别便是:CyclicBarrier可以重用,而CountDownLatch不行。CyclicBarrier中有个reset方法,如下:

返回值类型 方法名 方法描述
void reset() Resets the barrier to its initial state.

这个reset()方法的作用就是重置barrier到它的初始化状态。

CountDownLatch适用于一次性的事件,例如应用程序/模块 启动的时候。CyclicBarrier更适用于那种复发的事件场景。例如,输入数据不断改变时的事实计算。

join()方法有话要说

假设我们新建一个类,并在main函数中启动一个线程t1,然后调用t1.join().那么这个t1.join()方法就会是当前线程等待t1线程,直到t1线程执行结束。看到CountDownLatch,CyclicBarrier的作用之后,不禁要问,join方法也是促使线程等待的。那么join和上面说的二者又有什么区别呢?

CountDownLatch/CyclicBarrier并不要求所有工作线程执行结束,只要应用程序还在运行这些线程可以一直运行下去,也就是说“某一个工作”可以执行多次。此外,如果我们有N个任务,有M个线程,(N大于M)某线程可以执行多次任务,直到他们公共的Barrier N达到0。这个例子可以看出,CountDownLatch/CyclicBarrier更适用于让M个线程来共同完成一项工作的N个任务。

同时,在使用join()的时候,主线程需要在持有其他线程的引用的情况下,并调用join()方法。如果需要join的线程比较多,那么主线程就需要只有多个线程的引用(this is a bit dirty)。而在使用CountDownLatch/CyclicBarrier的时候,多个线程共享一个实例。(this is clear).

Java中CountDownLatch和CyclicBarrier的使用和比较的更多相关文章

  1. Java中CountDownLatch和CyclicBarrier

    Java编程思想中的例子import javax.validation.constraints.Size; import java.util.Random; import java.util.conc ...

  2. java并发编程中CountDownLatch和CyclicBarrier的使用

    在多线程程序设计中,经常会遇到一个线程等待一个或多个线程的场景,遇到这样的场景应该如何解决? 如果是一个线程等待一个线程,则可以通过await()和notify()来实现: 如果是一个线程等待多个线程 ...

  3. 使用Java辅助类(CountDownLatch、CyclicBarrier、Semaphore)并发编程

    在java 1.5中,提供了一些非常有用的辅助类来帮助我们进行并发编程,比如CountDownLatch,CyclicBarrier和Semaphore,今天我们就来学习一下这三个辅助类的用法 一.C ...

  4. java并发--CountDownLatch、CyclicBarrier和Semaphore

    在java 1.5中,提供了一些非常有用的辅助类来帮助我们进行并发编程,比如CountDownLatch,CyclicBarrier和Semaphore,今天我们就来学习一下这三个辅助类的用法. 以下 ...

  5. Java多线程-CountDownLatch、CyclicBarrier、Semaphore

    上次简单了解了多线程中锁的类型,今天要简单了解下多线程并发控制的一些工具类了. 1. 概念说明: CountDownLatch:相当于一个待执行线程计数器,当计数减为零时表示所有待执行线程都已执行完毕 ...

  6. Java的CountDownLatch和CyclicBarrier的理解和区别

    CountDownLatch和CyclicBarrier的功能看起来很相似,不易区分,有一种谜之的神秘.本文将通过通俗的例子并结合代码讲解两者的使用方法和区别. CountDownLatch和Cycl ...

  7. Java中CountDownLatch类的使用

    0.CountDownLatch作用 1) Java api中的解释:一个同步辅助类,在完成一组正在其他线程中执行的操作之前,它允许一个或多个线程一直等待. 2) CountDownLatch可以使J ...

  8. java面试-CountDownLatch、CyclicBarrier、Semaphore谈谈你的理解

    一.CountDownLatch 主要用来解决一个线程等待多个线程的场景,计数器不能循环利用 public class CountDownLatchDemo { public static void ...

  9. JAVA中CountDownLatch的简单示例

    public static void main(String[] args) throws InterruptedException { CountDownLatch latch =new Count ...

随机推荐

  1. 二、IIS部署WebApi

    一.项目发布 二.hosts 更改 C:\Windows\System32\drivers\etc 三.网站搭建 之后我将端口默认更改 8001   以防与80端口冲突 注意: 1.先测试IIS的lo ...

  2. 如何升级centos到最新版本

    本文将教你如何升级centos到最新版本.centos中“update”命令可以一次性更新所有软件到最新版本.注意:不推荐使用update的y选项,-y选项会让你在安装每项更新前都进行确认(译者注:这 ...

  3. ESXI 6.5安装详细步骤

    网址:http://blog.51cto.com/laotang6/2044861 ESXi是专为运行虚拟机.最大限度降低配置要求和简化部署而设计.只需几分钟时间,客户便可完成从安装到运行虚拟机的全过 ...

  4. 使用FFMPEG进行一些视频处理(C#)视频合并、转码、获取时长

    FFMPEG的强大无需多说,举几个用到的功能,直接贴代码了 还有更多命令用到时搜索即可 视频转码 ) { var args = "-y -i {0} -vcodec copy {1}&quo ...

  5. Makefile 使用总结(转)

    Makefile 使用总结  转自 https://www.cnblogs.com/wang_yb/p/3990952.html 1. Makefile 简介 Makefile 是和 make 命令一 ...

  6. crm 权限设计

    先在项目中创建  app rbac的models.py from django.db import models class Permission(models.Model): "" ...

  7. SEO学习知识

    监控流量的工具 百度统计 CNZZ  51LA 谷歌分析工具 如何从平台借流量? 竞价(付费).SEO 关键词定位: 定位人:负责人 将公司的业务全部列出来 选词: 根据定位的关键词选择出我们需要优化 ...

  8. VisualSVN服务器的本地搭建和使用

    Subversion是优秀的版本控制工具,其具体的的优点和详细介绍,这里就不再多说. 首先来下载和搭建SVN服务器. 现在Subversion已经迁移到apache网站上了,下载地址: http:// ...

  9. ETL过程跑完后,使用python发送邮件

    目标库中,如果有行数为0的表,使用python发送邮件 # -*- coding:utf-8 -*- # Author: zjc # Description:send monitor info to ...

  10. 应用留数定理计算实积分 $\dps{I(x)=\int_{-1}^1\frac{\rd t}{\sqrt{1-t^2}(t-x)}\ (|x|>1,x\in\bbR)}$ [华中师范大学2010年复变函数复试试题]

    应用留数定理计算实积分 $\dps{I(x)=\int_{-1}^1\frac{\rd t}{\sqrt{1-t^2}(t-x)}\ (|x|>1,x\in\bbR)}$ [华中师范大学2010 ...