java高级---->Thread之CountDownLatch的使用
CountDownLatch是JDK 5+里面闭锁的一个实现,允许一个或者多个线程等待某个事件的发生。今天我们通过一些实例来学习一下它的用法。
CountDownLatch的简单使用
CountDownLatch的简单介绍:
CountDownLatch类是一个同步计数器,构造时传入int参数,该参数就是计数器的初始值,每调用一次countDown()方法,计数器减1,计数器大于0 时,await()方法会阻塞程序继续执行
CountDownLatch是一个倒计数的锁存器,当计数减至0时触发特定的事件。
CountDownLatch被用来同步一个或多个任务,强制它们等待由其他任务执行的一组操作完成。你可以向CountDownLatch对象设置一个初始计数值,任何在这个对象上调用wait()方法都将阻塞,直至这个计数值达到0。其他任务在结束工作时,可以在该对象上调用凑你countDown()来减小这个计数值。CountDownLatch被设计为只触发一次,计数值不能被重置。如果你需要能够重置计数值的版本,则可以使用CyclicBarrier。
以下是jdk文档上的两个例子,我们可以学习一下。
一、开始信号,阻止任何工人进行,直到司机准备好他们继续进行
package com.linux.huhx.concurreny; import java.util.concurrent.CountDownLatch;
import java.util.concurrent.TimeUnit; public class Driver1 {
public static void main(String[] args) throws Exception {
CountDownLatch startSignal = new CountDownLatch(1);
CountDownLatch doneSignal = new CountDownLatch(5);
for (int i = 0; i < 5; ++i) // create and start threads
new Thread(new TaskWorker(startSignal, doneSignal)).start();
System.out.println("countDown before."); // don't let run yet
startSignal.countDown(); // let all threads proceed
TimeUnit.SECONDS.sleep(1);
System.out.println("await before");
/**
* 1、由于doneSignal里面的初始count=5,所以主线程在此会阻塞。
* 2、当所有的子线程执行完(doneSignal.countDown())之后,此时的count=0。所以后面代码可以继续执行。
*/
doneSignal.await();
System.out.println("await after");
} private static class TaskWorker implements Runnable {
private final CountDownLatch startSignal;
private final CountDownLatch doneSignal; public TaskWorker(CountDownLatch startSignal, CountDownLatch doneSignal) {
this.startSignal = startSignal;
this.doneSignal = doneSignal;
} @Override
public void run() {
try {
/**
* 1、刚进来的时候,由于startSignal的count=1。所以await()方法阻塞于此。
* 2、当主线程执行了startSignal.countDown();此时的startSignal的count=0。所以子线程可以正常运行。
*/
startSignal.await();
doSomeWork();
doneSignal.countDown();
} catch (InterruptedException e) {
e.printStackTrace();
}
} private void doSomeWork() {
System.out.println("in run method: " + Thread.currentThread().getName());
}
}
}
执行的结果如下:子线程的执行顺序不固定
countDown before.
in run method: Thread-
in run method: Thread-
in run method: Thread-
in run method: Thread-
in run method: Thread-
await before
await after
调用countDown()的任务在产生这个调用时并没有阻塞,只有对await()的调用会被阻塞,直至计数值到达0。
二、完成信号,允许司机等到所有工人完成为止
package com.linux.huhx.concurreny; import java.util.Random;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors; public class Driver2 {
public static void main(String[] args) throws Exception {
CountDownLatch doneSignal = new CountDownLatch(5);
ExecutorService service = Executors.newFixedThreadPool(3);
for (int i = 0; i < 5; ++i) // create and start threads
service.execute(new WorkerRunnable(doneSignal, i));
System.out.println("await before");
doneSignal.await();
System.out.println("await after");
service.shutdown();
} private static class WorkerRunnable implements Runnable {
private final CountDownLatch doneSignal;
private final int i; public WorkerRunnable(CountDownLatch doneSignal, int i) {
this.doneSignal = doneSignal;
this.i = i;
} @Override
public void run() {
doWork(i);
try {
Thread.sleep(new Random().nextInt(1000));
doneSignal.countDown();
} catch (InterruptedException e) {
e.printStackTrace();
}
} private void doWork(int i) {
System.out.println("in run method: " + i + " " + Thread.currentThread().getName());
}
}
}
运行的结果如下:子线程的执行顺序不固定
await before
in run method: pool--thread-
in run method: pool--thread-
in run method: pool--thread-
in run method: pool--thread-
in run method: pool--thread-
await after
友情链接
java高级---->Thread之CountDownLatch的使用的更多相关文章
- java高级---->Thread之CyclicBarrier的使用
CyclicBarrier是一个同步辅助类,它允许一组线程互相等待,直到到达某个公共屏障点 (common barrier point).今天我们就学习一下CyclicBarrier的用法. Cycl ...
- java高级---->Thread之ScheduledExecutorService的使用
ScheduledExecutorService的主要作用就是可以将定时任务与线程池功能结合使用.今天我们来学习一下ScheduledExecutorService的用法.我们都太渺小了,那么容易便湮 ...
- java高级---->Thread之ExecutorService的使用
今天我们通过实例来学习一下ExecutorService的用法.我徒然学会了抗拒热闹,却还来不及透悟真正的冷清. ExecutorService的简单实例 一.ExecutorService的简单使用 ...
- java高级---->Thread之Phaser的使用
Phaser提供了动态增parties计数,这点比CyclicBarrier类操作parties更加方便.它是jdk1.7新增的类,今天我们就来学习一下它的用法.尘埃落定之后,回忆别来挑拨. Phas ...
- java高级---->Thread之CompletionService的使用
CompletionService的功能是以异步的方式一边生产新的任务,一边处理已完成任务的结果,这样可以将执行任务与处理任务分离开来进行处理.今天我们通过实例来学习一下CompletionServi ...
- java高级---->Thread之BlockingQueue的使用
今天我们通过实例来学习一下BlockingQueue的用法.梦想,可以天花乱坠,理想,是我们一步一个脚印踩出来的坎坷道路. BlockingQueue的实例 官方文档上的对于BlockingQueue ...
- java高级---->Thread之Exchanger的使用
Exchanger可以在两个线程之间交换数据,只能是2个线程,他不支持更多的线程之间互换数据.今天我们就通过实例来学习一下Exchanger的用法. Exchanger的简单实例 Exchanger是 ...
- java高级---->Thread之FutureTask的使用
FutureTask类是Future 的一个实现,并实现了Runnable,所以可通过Excutor(线程池) 来执行,也可传递给Thread对象执行.今天我们通过实例来学习一下FutureTask的 ...
- java高级---->Thread之Condition的使用
Condition 将 Object 监视器方法(wait.notify 和 notifyAll)分解成截然不同的对象,以便通过将这些对象与任意 Lock 实现组合使用,为每个对象提供多个等待 set ...
随机推荐
- BusyBox inittab配置
inittab第一行指定初始化脚本,开启所有应用程序,必须有. ::sysinit:/etc/init.d/rcS 在第一行执行完后,执行剩余行命令,此终端要接受命令交互需开启sh. ::askfir ...
- scsi线
- Aria2 Axel wget curl 四个下载命令
经常由于不同需求使用下载管理器从互联网下载文件,它给我和其他人提供了很多帮助.我们都想要一个超级快速的下载管理器来完成下载尽可能多的任务,以便我们可以节省时间来进一步地工作.有很多可以加速下载的下载管 ...
- expect模块的使用,主要没装包折腾一晚上
第一步首先下载expect 模块,yum list |grep expect ,安装下面的模块. expect.x86_64 5.44.1. ...
- node-webkit播放目录下所有网页文件
1.编写index.html文件 这里我们需要播放某一个目录下所有的网页文件,要做到切换不同的网页,可以在index.html中使用iframe.通过js动态改变iframe的src属性,从而动态地切 ...
- Ubuntu之Docker安装
1.添加官方的GPG key到系统 curl -fsSL https://download.docker.com/linux/ubuntu/gpg | sudo apt-key add - 2.添加D ...
- 使用rcp命令实现从远程机器拷贝文件
rcp:远程文件复制 使用rcp命令可以在远程主机之间复制文件,如果同时指定2个以上的文件或目录,且最后的目的地是一个已经存在的目录,则他会把前面指定的所有文件或目录复制到该目录中. rcp命令语法: ...
- 【转】【Java/Android】Toast使用方法大全
Toast 是一个 View 视图,快速的为用户显示少量的信息. Toast 在应用程序上浮动显示信息给用户,它永远不会获得焦点,不影响用户的输入等操作,主要用于 一些帮助 / 提示.Toast 最常 ...
- Spring零散所得
Spring容器中bean的id或name,都可以有多个,且第一个为标识符(Qualifier),其余皆为别名(Alias).所以都可以通过applicationContext.getBean(&qu ...
- 无法从“重载函数类型”为“const std::_Tree<_Traits> &”推导 <未知> 参数
场景: 原因: 用到string类型,但是没有包含头文件. 解决方法: #include<string>