- 浏览: 228721 次
- 性别:
- 来自: 上海
文章分类
- 全部博客 (102)
- 开源软件 (1)
- 并发 (14)
- WEB (1)
- NIO (4)
- Socket (5)
- 应用服务器 (4)
- 集群 (0)
- 数据库 (1)
- JAVA基础 (17)
- 开源框架 (2)
- 业务知识 (1)
- JVM (9)
- Windows (1)
- LINUX (0)
- Jquery (0)
- JMS (0)
- Cache (0)
- Oracle (5)
- XML (0)
- EJB (0)
- WebService (0)
- Struts2 (1)
- Hibernate (1)
- Spring (0)
- 设计模式 (4)
- UML (0)
- JS (12)
- 网络爬虫 (0)
- 数据结构与算法 (1)
- EXT (1)
- DIV+CSS (2)
- 安全 (3)
- Android (9)
- LDAP (1)
- Mybatis (1)
最新评论
-
Dom_4j:
...
理解注解中的@Inherited -
s469799470:
demo少个ID
iframe父子页面交互问题 -
errorerror0:
...
iframe父子页面交互问题 -
errorerror0:
iframe父子页面交互问题 -
johnawm:
2012-12-18 wangshibei 写道CountD ...
CountDownLatch的使用
concurrent包里面的CountDownLatch其实可以把它看作一个计数器,只不过这个计数器的操作是原子操作,同时只能有一个线程去操作这个计数器,也就是同时只能有一个线程去减这个计数器里面的值。
CountDownLatch的一个非常典型的应用场景是:有一个任务想要往下执行,但必须要等到其他的任务执行完毕后才可以继续往下执行。假如我们这个想要继续往下执行的任务调用一个CountDownLatch对象的await()方法,其他的任务执行完自己的任务后调用同一个CountDownLatch对象上的countDown()方法,这个调用await()方法的任务将一直阻塞等待,直到这个CountDownLatch对象的计数值减到0为止。
举个例子,有三个工人在为老板干活,这个老板有一个习惯,就是当三个工人把一天的活都干完了的时候,他就来检查所有工人所干的活。记住这个条件:三个工人先全部干完活,老板才检查。所以在这里用Java代码设计两个类,Worker代表工人,Boss代表老板,具体的代码实现如下:
public class Worker implements Runnable{
private CountDownLatch downLatch;
private String name;
public Worker(CountDownLatch downLatch, String name){
this.downLatch = downLatch;
this.name = name;
}
public void run() {
this.doWork();
try
{
TimeUnit.SECONDS.sleep(new Random().nextInt(10));
}catch(InterruptedException ie){
}
System.out.println(this.name + "活干完了!");
this.downLatch.countDown();
}
private void doWork()
{
System.out.println(this.name + "正在干活!");
}
}
==================
public class Boss implements Runnable {
private CountDownLatch downLatch;
public Boss(CountDownLatch downLatch){
this.downLatch = downLatch;
}
public void run() {
System.out.println("老板正在等所有的工人干完活......");
try {
this.downLatch.await();
} catch (InterruptedException e) {
}
System.out.println("工人活都干完了,老板开始检查了!");
}
}
============================
public class CountDownLatchDemo {
public static void main(String[] args) {
ExecutorService executor = Executors.newCachedThreadPool();
CountDownLatch latch = new CountDownLatch(3);
Worker w1 = new Worker(latch,"张三");
Worker w2 = new Worker(latch,"李四");
Worker w3 = new Worker(latch,"王二");
Boss boss = new Boss(latch);
executor.execute(w3);
executor.execute(w2);
executor.execute(w1);
executor.execute(boss);
executor.shutdown();
}
}
使用System.nanoTime()试试吧...
System.out.println()是main的线程 不受ExecutorService 管理,他们互补干扰,
使用System.nanoTime()试试吧...
CountDownLatch的一个非常典型的应用场景是:有一个任务想要往下执行,但必须要等到其他的任务执行完毕后才可以继续往下执行。假如我们这个想要继续往下执行的任务调用一个CountDownLatch对象的await()方法,其他的任务执行完自己的任务后调用同一个CountDownLatch对象上的countDown()方法,这个调用await()方法的任务将一直阻塞等待,直到这个CountDownLatch对象的计数值减到0为止。
举个例子,有三个工人在为老板干活,这个老板有一个习惯,就是当三个工人把一天的活都干完了的时候,他就来检查所有工人所干的活。记住这个条件:三个工人先全部干完活,老板才检查。所以在这里用Java代码设计两个类,Worker代表工人,Boss代表老板,具体的代码实现如下:
public class Worker implements Runnable{
private CountDownLatch downLatch;
private String name;
public Worker(CountDownLatch downLatch, String name){
this.downLatch = downLatch;
this.name = name;
}
public void run() {
this.doWork();
try
{
TimeUnit.SECONDS.sleep(new Random().nextInt(10));
}catch(InterruptedException ie){
}
System.out.println(this.name + "活干完了!");
this.downLatch.countDown();
}
private void doWork()
{
System.out.println(this.name + "正在干活!");
}
}
==================
public class Boss implements Runnable {
private CountDownLatch downLatch;
public Boss(CountDownLatch downLatch){
this.downLatch = downLatch;
}
public void run() {
System.out.println("老板正在等所有的工人干完活......");
try {
this.downLatch.await();
} catch (InterruptedException e) {
}
System.out.println("工人活都干完了,老板开始检查了!");
}
}
============================
public class CountDownLatchDemo {
public static void main(String[] args) {
ExecutorService executor = Executors.newCachedThreadPool();
CountDownLatch latch = new CountDownLatch(3);
Worker w1 = new Worker(latch,"张三");
Worker w2 = new Worker(latch,"李四");
Worker w3 = new Worker(latch,"王二");
Boss boss = new Boss(latch);
executor.execute(w3);
executor.execute(w2);
executor.execute(w1);
executor.execute(boss);
executor.shutdown();
}
}
评论
6 楼
johnawm
2013-01-05
2012-12-18
有道理!!
wangshibei 写道
CountDownLatch这个东西还有点用,当计数器减为0时调用的应该是“notifyall”,而不是“notify”,所以只需要“latch”就可以了,不用“doneLatch ”
johnawm 写道
要统计时间你可以再加一个CountDownLatch变量用来表示老板检查结束,如
package methodtest; import java.util.Random; import java.util.concurrent.CountDownLatch; import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; import java.util.concurrent.TimeUnit; public class CountDonwLatchTest { public static void main(String[] a) { long startTime = System.currentTimeMillis(); ExecutorService executor = Executors.newCachedThreadPool(); CountDownLatch latch = new CountDownLatch(3); CountDownLatch doneLatch = new CountDownLatch(1); Worker w1 = new Worker(latch, "张三"); Worker w2 = new Worker(latch, "李四"); Worker w3 = new Worker(latch, "王二"); Boss boss = new Boss(latch, doneLatch); executor.execute(w3); executor.execute(w2); executor.execute(w1); executor.execute(boss); executor.shutdown(); try { doneLatch.await(); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println(System.currentTimeMillis() - startTime); } } class Worker implements Runnable { private CountDownLatch downLatch; private String name; public Worker(CountDownLatch downLatch, String name) { this.downLatch = downLatch; this.name = name; } public void run() { this.doWork(); try { TimeUnit.SECONDS.sleep(new Random().nextInt(10)); } catch (InterruptedException ie) { } System.out.println(this.name + "活干完了!"); this.downLatch.countDown(); } private void doWork() { System.out.println(this.name + "正在干活!"); } } class Boss implements Runnable { private CountDownLatch downLatch; private CountDownLatch doneLatch; public Boss(CountDownLatch downLatch,CountDownLatch doneLatch) { this.downLatch = downLatch; this.doneLatch = doneLatch; } public void run() { System.out.println("老板正在等所有的工人干完活..."); try { this.downLatch.await(); } catch (InterruptedException e) { } System.out.println("工人活都干完了,老板开始检查了!"); doneLatch.countDown(); } }
有道理!!
5 楼
wangshibei
2012-12-18
CountDownLatch这个东西还有点用,当计数器减为0时调用的应该是“notifyall”,而不是“notify”,所以只需要“latch”就可以了,不用“doneLatch ”
johnawm 写道
要统计时间你可以再加一个CountDownLatch变量用来表示老板检查结束,如
package methodtest; import java.util.Random; import java.util.concurrent.CountDownLatch; import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; import java.util.concurrent.TimeUnit; public class CountDonwLatchTest { public static void main(String[] a) { long startTime = System.currentTimeMillis(); ExecutorService executor = Executors.newCachedThreadPool(); CountDownLatch latch = new CountDownLatch(3); CountDownLatch doneLatch = new CountDownLatch(1); Worker w1 = new Worker(latch, "张三"); Worker w2 = new Worker(latch, "李四"); Worker w3 = new Worker(latch, "王二"); Boss boss = new Boss(latch, doneLatch); executor.execute(w3); executor.execute(w2); executor.execute(w1); executor.execute(boss); executor.shutdown(); try { doneLatch.await(); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println(System.currentTimeMillis() - startTime); } } class Worker implements Runnable { private CountDownLatch downLatch; private String name; public Worker(CountDownLatch downLatch, String name) { this.downLatch = downLatch; this.name = name; } public void run() { this.doWork(); try { TimeUnit.SECONDS.sleep(new Random().nextInt(10)); } catch (InterruptedException ie) { } System.out.println(this.name + "活干完了!"); this.downLatch.countDown(); } private void doWork() { System.out.println(this.name + "正在干活!"); } } class Boss implements Runnable { private CountDownLatch downLatch; private CountDownLatch doneLatch; public Boss(CountDownLatch downLatch,CountDownLatch doneLatch) { this.downLatch = downLatch; this.doneLatch = doneLatch; } public void run() { System.out.println("老板正在等所有的工人干完活..."); try { this.downLatch.await(); } catch (InterruptedException e) { } System.out.println("工人活都干完了,老板开始检查了!"); doneLatch.countDown(); } }
4 楼
johnawm
2012-08-28
要统计时间你可以再加一个CountDownLatch变量用来表示老板检查结束,如
package methodtest; import java.util.Random; import java.util.concurrent.CountDownLatch; import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; import java.util.concurrent.TimeUnit; public class CountDonwLatchTest { public static void main(String[] a) { long startTime = System.currentTimeMillis(); ExecutorService executor = Executors.newCachedThreadPool(); CountDownLatch latch = new CountDownLatch(3); CountDownLatch doneLatch = new CountDownLatch(1); Worker w1 = new Worker(latch, "张三"); Worker w2 = new Worker(latch, "李四"); Worker w3 = new Worker(latch, "王二"); Boss boss = new Boss(latch, doneLatch); executor.execute(w3); executor.execute(w2); executor.execute(w1); executor.execute(boss); executor.shutdown(); try { doneLatch.await(); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println(System.currentTimeMillis() - startTime); } } class Worker implements Runnable { private CountDownLatch downLatch; private String name; public Worker(CountDownLatch downLatch, String name) { this.downLatch = downLatch; this.name = name; } public void run() { this.doWork(); try { TimeUnit.SECONDS.sleep(new Random().nextInt(10)); } catch (InterruptedException ie) { } System.out.println(this.name + "活干完了!"); this.downLatch.countDown(); } private void doWork() { System.out.println(this.name + "正在干活!"); } } class Boss implements Runnable { private CountDownLatch downLatch; private CountDownLatch doneLatch; public Boss(CountDownLatch downLatch,CountDownLatch doneLatch) { this.downLatch = downLatch; this.doneLatch = doneLatch; } public void run() { System.out.println("老板正在等所有的工人干完活..."); try { this.downLatch.await(); } catch (InterruptedException e) { } System.out.println("工人活都干完了,老板开始检查了!"); doneLatch.countDown(); } }
3 楼
bubiaiyou
2012-08-27
javac_xinyun 写道
somefuture 写道
如果想统计时间,我这样做
long start871456 = System.currentTimeMillis();
executor.execute(w3);
executor.execute(w2);
executor.execute(w1);
executor.execute(boss);
executor.shutdown();
long end84654856 = System.currentTimeMillis();
System.out.println("运行时间是" + (end84654856 - start871456) + "毫秒");
为什么不行呢?
long start871456 = System.currentTimeMillis();
executor.execute(w3);
executor.execute(w2);
executor.execute(w1);
executor.execute(boss);
executor.shutdown();
long end84654856 = System.currentTimeMillis();
System.out.println("运行时间是" + (end84654856 - start871456) + "毫秒");
为什么不行呢?
使用System.nanoTime()试试吧...
System.out.println()是main的线程 不受ExecutorService 管理,他们互补干扰,
2 楼
javac_xinyun
2012-08-09
somefuture 写道
如果想统计时间,我这样做
long start871456 = System.currentTimeMillis();
executor.execute(w3);
executor.execute(w2);
executor.execute(w1);
executor.execute(boss);
executor.shutdown();
long end84654856 = System.currentTimeMillis();
System.out.println("运行时间是" + (end84654856 - start871456) + "毫秒");
为什么不行呢?
long start871456 = System.currentTimeMillis();
executor.execute(w3);
executor.execute(w2);
executor.execute(w1);
executor.execute(boss);
executor.shutdown();
long end84654856 = System.currentTimeMillis();
System.out.println("运行时间是" + (end84654856 - start871456) + "毫秒");
为什么不行呢?
使用System.nanoTime()试试吧...
1 楼
somefuture
2012-08-03
如果想统计时间,我这样做
long start871456 = System.currentTimeMillis();
executor.execute(w3);
executor.execute(w2);
executor.execute(w1);
executor.execute(boss);
executor.shutdown();
long end84654856 = System.currentTimeMillis();
System.out.println("运行时间是" + (end84654856 - start871456) + "毫秒");
为什么不行呢?
long start871456 = System.currentTimeMillis();
executor.execute(w3);
executor.execute(w2);
executor.execute(w1);
executor.execute(boss);
executor.shutdown();
long end84654856 = System.currentTimeMillis();
System.out.println("运行时间是" + (end84654856 - start871456) + "毫秒");
为什么不行呢?
发表评论
-
线程中断
2012-11-15 16:04 3196线程的中断"interrupt& ... -
任务间使用管道输入输出
2011-10-17 10:42 1118管道流PipedReader和PipedWriter,它不同于 ... -
使用同步队列解决任务协作问题
2011-10-17 10:17 1393Java API里面提供了许多同步队列,你可以使用它来解决任务 ... -
简单生产者-消费者
2011-10-16 22:46 982考虑一个场景:一个饭店有一个厨师和一个服务员。菜做好之后,厨师 ... -
NIO响应中断
2012-11-30 15:59 1048不同于IO,NIO通道会自动地相应中断,代码如下: publi ... -
关闭底层资源强行中断
2011-10-11 10:26 1252对于IO这种资源无法通过interrupt()方法中断,但是在 ... -
线程中断
2011-10-11 10:09 909线程类的interrupt()方法可对某一线程进行中断,例子如 ... -
多线程中的原子类
2011-10-11 09:47 2046JAVA提供了AtomicInteger,AtomicLong ... -
多个任务运行终止以及资源共享例子
2011-10-11 09:27 929场景: 公园的管理人员 ... -
不正确的访问共享资源
2011-10-10 22:06 873看下面一段代码,这段 ... -
从任务中产生返回值
2011-10-10 21:44 929Runnable接口不返回任何值,如果希望任务完成时能返回一个 ... -
学会显示地使用Lock
2011-10-10 19:41 975Lock对象在使用的时候必须被显示地创建、锁定和释放。它和sy ... -
Daemon的使用
2011-10-10 19:12 1016后台线程是程序在运行的时候在后台提供一种通用服务的线程,当所有 ...
相关推荐
主要介绍了JAVA多线程CountDownLatch的使用,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
文章目录1 原理简介2 具体使用方法2.1 demo1 — await不传入时间,保证当前线程的其他操作在最后执行2.2 demo2 — await传入时间t,当前线程等其他线程时间t后就运行其他操作2.3 发令枪 源码地址:...
CountDownLatch使用实例 - 初始化时,设置计数(count)值,也就是闭锁需要等待的线程数。 - 主线程必须在启动其他线程后立即调用 CountDownLatch.await() 方法,这样主线程的操作就会在这个方法上阻塞,直到其他...
于是乎到现在的Hibernate、MyBatis、Spring、Spring MVC、AQS、ThreadPoolExecutor、CountDownLatch使用场景和核心源码分析。 感觉自己还是真的菜鸡,有太多框架的底层实现都不怎么了解。 当山头被一座一座攻克时,...
java并发编程中CountDownLatch和CyclicBarrier的使用借鉴.pdf
《java并发编程》中CountDownLatch和CyclicBarrier用法实例大全,几乎包含了所有重要的用法
利用 CountDownLatch 类实现线程同步,而不用回调机制。详见我的博文 http://blog.csdn.net/kroclin/article/details/37956949
mybaits 多线程 实现数据批量插入 (运用CountDownLatch实现闭锁) 1、mybatis批处理 2、数据分批量查询 3、数据分批量插入
目录 CountDownLatch是什么? CountDownLatch如何工作? 在实时系统中的应用场景 应用范例 常见的面试题 代码样例
CountDownLatch与thread.join()的区别
在网上找的一个CountDownLatch的学习demo,感觉很不错,就摘抄过来了
java并发编程中CountDownLatch和CyclicBarrier的使用.pdf
NULL 博文链接:https://cpjsjxy.iteye.com/blog/2272451
CountDownLatch Demo
Java并发编程一CountDownLatch、CyclicBarrier、Semaphore初使用 CountDownLatch、CyclicBarrier、Semaphore这些线程协作工具类是基于AQS的,看完这篇博客后可以去看下面这篇博客,了解它们是如何实现的。 Java并发...
并发编程之CountDownLatch
3.2 使用CountDownLatch实现同步 主线程等待多个线程完成 4.1 场景介绍 4.2 使用CountDownLatch实现等待 CountDownLatch的其他应用场景 5.1 倒计时计时器 5.2 同时开始任务 5.3 等待多个资源就绪 CountDownLatch与...
主要为大家详细介绍了CountDownLatch的使用说明,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
主要介绍了如何使用CountDownLatch同步java多线程,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下