多线程12-CyclicBarrier、CountDownLatch、Exchanger
1.CyclicBarrier
表示大家彼此等待,大家集合好后才开始出发,分散活动后又在指定地点集合碰面
package org.lkl.thead.foo; import java.util.concurrent.CyclicBarrier;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors; /**
* 表示大家彼此等待,大家集合好后才开始出发,分散活动后又在指定地点集合碰面
*/
public class CyclicBarrierFoo {
public static void main(String[] args) {
ExecutorService threadPool = Executors.newCachedThreadPool() ; final CyclicBarrier cyclicBarrier = new CyclicBarrier() ; //表示有3个线程需要彼此等待 都执行到 cyclicBarrier.await() ;以后才继续执行 for(int i= ;i<= ;i++){
Runnable r = new Runnable() {
@Override
public void run() { try {
Thread.sleep((long)(Math.random()*));
System.out.println("线程" + Thread.currentThread().getName() +
"即将到达集合地点1,当前已有" + (cyclicBarrier.getNumberWaiting()+) + "个已经到达," + (cyclicBarrier.getNumberWaiting()==?"都到齐了,继续走啊":"正在等候")); cyclicBarrier.await() ; //类似于集合点 Thread.sleep((long)(Math.random()*));
System.out.println("线程" + Thread.currentThread().getName() +
"即将到达集合地点2,当前已有" + (cyclicBarrier.getNumberWaiting()+) + "个已经到达," + (cyclicBarrier.getNumberWaiting()==?"都到齐了,继续走啊":"正在等候")); cyclicBarrier.await() ; //类似于集合点 Thread.sleep((long)(Math.random()*));
System.out.println("线程" + Thread.currentThread().getName() +
"即将到达集合地点3,当前已有" + (cyclicBarrier.getNumberWaiting()+) + "个已经到达," + (cyclicBarrier.getNumberWaiting()==?"都到齐了,继续走啊":"正在等候")); cyclicBarrier.await() ; //类似于集合点 } catch (Exception e) {
e.printStackTrace();
}
}
};
threadPool.execute(r) ;
}
}
}
允许结果如下:
线程pool--thread-3即将到达集合地点1,当前已有1个已经到达,正在等候
线程pool--thread-1即将到达集合地点1,当前已有2个已经到达,正在等候
线程pool--thread-2即将到达集合地点1,当前已有3个已经到达,都到齐了,继续走啊
线程pool--thread-3即将到达集合地点2,当前已有1个已经到达,正在等候
线程pool--thread-1即将到达集合地点2,当前已有2个已经到达,正在等候
线程pool--thread-2即将到达集合地点2,当前已有3个已经到达,都到齐了,继续走啊
线程pool--thread-2即将到达集合地点3,当前已有1个已经到达,正在等候
线程pool--thread-1即将到达集合地点3,当前已有2个已经到达,正在等候
线程pool--thread-3即将到达集合地点3,当前已有3个已经到达,都到齐了,继续走啊
2. CountdownLatchFoo
犹如倒计时计数器,调用CountDownLatch对象的countDown方法就将计数器减1,当计数到达0时,则所有等待者或单个等待者开始执行
可以实现一个人(也可以是多个人)等待其他所有人都来通知他,这犹如一个计划需要多个领导都签字后才能继续向下实施。还可以实现一个人通知多个人的效果,类似裁判一声口令,运动员同时开始奔跑
package org.lkl.thead.foo; import java.util.concurrent.CountDownLatch;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors; public class CountdownLatchFoo { /**
* 实现代码:
* 主线程发布命令 然后开启三个线程在等待主线程发布的命令 ,
* 当子线程接收命令以后 传递一个回调的命令给主线程 然后主线程接收回调命令
*/ public static void main(String[] args) {
final CountDownLatch orderLatch = new CountDownLatch() ; //表示从1 开始倒计时
final CountDownLatch answerLatch = new CountDownLatch() ; //表示从3开始倒计时 ExecutorService threadPool = Executors.newCachedThreadPool() ;
for(int i= ;i<= ;i++){
Runnable runnable = new Runnable() {
@Override
public void run() {
System.out.println("线程"+Thread.currentThread().getName()+"准备接收数据.");
try {
orderLatch.await() ; //表示等待主线程发布命令,没有接收到命令就会一致等待下去
} catch (InterruptedException e) {
e.printStackTrace();
} System.out.println("线程"+Thread.currentThread().getName()+" 接收到命令 "); //只有接收到命令 程序才会从await方法中执行下来 //给主线程回调命令 System.out.println("线程"+Thread.currentThread().getName()+" 准备向主线程回调. ");
try {
Thread.sleep((long)(Math.random()*));
} catch (InterruptedException e) {
e.printStackTrace();
}
answerLatch.countDown() ; //表示发布命令 从3开始倒计时 前期主线程会被answerLatch.await() 方法拦截下来 等待回调 System.out.println("线程"+Thread.currentThread().getName()+" 已经向主线程回调. "); }
};
threadPool.execute(runnable) ;
} //主线程 try {
Thread.sleep((long)(Math.random()*));
System.out.println("主线程"+Thread.currentThread().getName()+"准备向三个子线程发送命令.");
orderLatch.countDown() ; //向三个子线程发送命令
System.out.println("主线程"+Thread.currentThread().getName()+"已经向三个子线程发送命令."); System.out.println("主线程"+Thread.currentThread().getName()+"等待三个子线程的回调.");
answerLatch.await() ; System.out.println("主线程"+Thread.currentThread().getName()+"已经收到三个子线程的回调.");
} catch (Exception e) {
e.printStackTrace();
}
}
}
运行结果:
线程pool--thread-1准备接收数据.
线程pool--thread-2准备接收数据.
线程pool--thread-3准备接收数据.
主线程main准备向三个子线程发送命令.
线程pool--thread- 接收到命令
线程pool--thread- 准备向主线程回调.
主线程main已经向三个子线程发送命令.
主线程main等待三个子线程的回调.
线程pool--thread- 接收到命令
线程pool--thread- 准备向主线程回调.
线程pool--thread- 接收到命令
线程pool--thread- 准备向主线程回调.
线程pool--thread- 已经向主线程回调.
线程pool--thread- 已经向主线程回调.
线程pool--thread- 已经向主线程回调.
主线程main已经收到三个子线程的回调.
3. Exchanger
用于实现两个人之间的数据交换,每个人在完成一定的事务后想与对方交换数据,第一个先拿出数据的人将一直等待第二个人拿着数据到来时,才能彼此交换数据
package org.lkl.thead.foo; import java.util.concurrent.Exchanger;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors; public class ExchangerFoo { public static void main(String[] args) {
ExecutorService threadPool = Executors.newCachedThreadPool() ;
final Exchanger<String> change = new Exchanger<String>() ;
threadPool.execute(new Runnable() { @Override
public void run() {
try {
String myData = "zhangsan" ;
System.out.println("线程" + Thread.currentThread().getName() +
"正在把数据" + myData +"换出去");
String mychange = change.exchange(myData) ;
Thread.sleep((long)(Math.random()*));
System.out.println("线程" + Thread.currentThread().getName() +
"换回来的数据为" + mychange);
} catch (Exception e) {
// TODO Auto-generated catch block
e.printStackTrace();
} }
}) ; threadPool.execute(new Runnable() { @Override
public void run() {
try {
String myData = "lisi" ;
System.out.println("线程" + Thread.currentThread().getName() +
"正在把数据" + myData +"换出去");
String mychange = change.exchange(myData) ;
Thread.sleep((long)(Math.random()*));
System.out.println("线程" + Thread.currentThread().getName() +
"换回来的数据为" + mychange);
} catch (Exception e) {
// TODO Auto-generated catch block
e.printStackTrace();
} }
}) ; }
}
运行结果:
线程pool--thread-2正在把数据lisi换出去
线程pool--thread-1正在把数据zhangsan换出去
线程pool--thread-2换回来的数据为zhangsan
线程pool--thread-1换回来的数据为lisi
多线程12-CyclicBarrier、CountDownLatch、Exchanger的更多相关文章
- Java中的4个并发工具类 CountDownLatch CyclicBarrier Semaphore Exchanger
在 java.util.concurrent 包中提供了 4 个有用的并发工具类 CountDownLatch 允许一个或多个线程等待其他线程完成操作,课题点 Thread 类的 join() 方法 ...
- java多线程系列(八)---CountDownLatch和CyclicBarrie
CountDownLatch 前言:如有不正确的地方,还望指正. 目录 认识cpu.核心与线程 java多线程系列(一)之java多线程技能 java多线程系列(二)之对象变量的并发访问 java多线 ...
- 30行自己写并发工具类(Semaphore, CyclicBarrier, CountDownLatch)是什么体验?
30行自己写并发工具类(Semaphore, CyclicBarrier, CountDownLatch)是什么体验? 前言 在本篇文章当中首先给大家介绍三个工具Semaphore, CyclicBa ...
- Java_并发线程_Semaphore、CountDownLatch、CyclicBarrier、Exchanger
1.Semaphore 信号量(Semaphore),有时被称为信号灯,是在多线程环境下使用的一种设施, 它负责协调各个线程, 以保证它们可以正确.合理的使用公共资源. Semaphore当前在多线程 ...
- Java核心知识点学习----多线程 倒计时记数器CountDownLatch和数据交换的Exchanger
本文将要介绍的内容都是Java5中的新特性,一个是倒计时记数器---CountDownLatch,另一个是用于线程间数据交换的Exchanger. 一.CountDownLatch 1.什么是Coun ...
- 并发工具类的使用 CountDownLatch,CyclicBarrier,Semaphore,Exchanger
1.CountDownLatch 允许一个或多个线程等待直到在其他线程中执行的一组操作完成的同步辅助. A CountDownLatch用给定的计数初始化. await方法阻塞,直到由于countDo ...
- 温故知新-多线程-forkjoin、CountDownLatch、CyclicBarrier、Semaphore用法
Posted by 微博@Yangsc_o 原创文章,版权声明:自由转载-非商用-非衍生-保持署名 | Creative Commons BY-NC-ND 3.0 文章目录 摘要 forkjoin C ...
- Java多线程中对CountDownLatch的使用
CountDownLatch是一个非常实用的多线程控制工具类,称之为“倒计时器”,它允许一个或多个线程一直等待,直到其他线程的操作执行完后再执行.用给定的计数初始化CountDownLatch,其含义 ...
- JUC并发工具包之CyclicBarrier & CountDownLatch的异同
1.介绍 本文我们将比较一下CyclicBarrier和CountDownLatch并了解两者的相似与不同. 2.两者是什么 当谈到并发,将这两者概念化的去解释两者是做什么的,这其实是一件很有挑战的事 ...
随机推荐
- 简单又强大的联发科手机PhilZ Touch Recovery安装器,详细教程 - 本文出自高州吧
原文地址:http://bbs.gaozhouba.com/thread-19355-1-1.html * * * * * * * * * * * * * * * * * * * * * * * * ...
- 关于KeilC51的指针(参见, page 106-113, keil uv2 user's guide 09,2001)
keil中的指针分为两种,一种是普通指针,兼容标准C语言的指针:另一种是我翻译成内存特殊指针(memory-specific pointers,翻译的不好:>) 一.普通指针 普通指针的定义方式 ...
- COJ 0503 比赛
比赛 难度级别:D: 运行时间限制:2000ms: 运行空间限制:51200KB: 代码长度限制:2000000B 试题描述 初三年级举办了一场篮球赛,共有N个班级参加.当WZJ知道了这件事情, 已经 ...
- 将多个图片整合到一张图片中再用CSS 进行网页背景定位
原文地址:http://wenku.baidu.com/link?url=hj_qM9kmdMrg8KWXFD2bCF_uuJCxKJRvG97CkWk3itsPq3izMzfrKvSZYBzDGyP ...
- c宏的MAX函数
今天从香山上面回来累的跟傻逼一样,回来问了一下胡总的阿里面试的问题.然后其中有一个是宏写max函数.胡总说不好写,然后我就去洗澡了. 洗澡的时候感觉不对啊,回来写了一个: #define MAX(a, ...
- Diamond Armor - The most expensive Suit: 2.8 Mio Swiss Francs
Diamond Armor - The most expensive Suit: 2.8 Mio Swiss Francs Diamond Armor
- oracle管道输出
通常我们会在oracle中用dbms_output输出调试信息,但dbms_output只能在调用过程完成才返回结果,不能实时输出的.这意味着通常我们经常要等几分钟或更长的时间才能看到调试信息,那怎么 ...
- swift - use backslash to add the value in the string
There’s an even simpler way to include values in strings: Write the value in parentheses, and write ...
- nexus私服安装
一.搭建nexus私服.当前服务器版本是jdk1.8 . nexus安装包下载:http://www.sonatype.org/nexus/archived 先是下载目前最新的版本 Nexus ...
- android:ellipsize的使用
EidtText和textview中内容过长的话自动换行,使用android:ellipsize与android:singleine可以解决,使只有一行. EditText不支持marquee 用法如 ...