多线程003 - 再谈CyclicBarrier
java.util.concurrent.CyclicBarrier也是JDK 1.5提供的一个同步辅助类(为什么用也呢?參见再谈CountDownLatch)。它同意一组线程互相等待,直到到达某个临界点(a common barrier point,翻译成公共障碍点、公共栅栏点都不够传神,直接用临界点吧)。在某个程序中,一组固定大小的线程必须互相等待时。CyclicBarrier将起非常大的作用。由于在等待线程被释放后,这个临界点能够重用。所以说是循环的。
CyclicBarrier支持一个可选的Runnable。在一组线程中的最后一个线程完毕之后、释放全部线程之前,该Runnable在屏障点执行一次(每循环一次Runnable执行一次)。这样的方式能够用来在下一波继续执行的线程执行之前更新共享状态(比方下一波僵尸来之前。检查武器弹药)。
CountDownLatch与CyclicBarrier
CountDownLatch是不可以反复使用的。是一次性的,其锁定一经打开。就不可以在反复使用。
就像引线。点燃后就在燃烧降低。燃烧完了就不能再次使用了。
CyclicBarrier是一种循环的方式进行锁定,这次锁定被打开之后,还可以反复计数。再次使用。就像沙漏。这次漏完了。倒过来接着漏。
另一点是两者之间非常大的差别,就是CountDownLatch在等待子线程的过程中,会锁定主线程,而CyclicBarrier不会锁定主线程,仅仅是在全部子线程结束后。依据定义运行其可选的Runnable线程。
所以在这两种辅助类中进行选择时,可以非常明显进行区分。
CyclicBarrier实例
能够考虑这么一种情况,我们须要向数据库导入一些数据,没导入几条希望能进行一次计时,便于我们查看。由于实现比較简单,直接上代码:
package howe.demo.thread.cyclicbarrier; import java.util.Random;
import java.util.concurrent.BrokenBarrierException;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.CyclicBarrier;
import java.util.concurrent.TimeUnit; /**
* @author liuxinghao
* @version 1.0 Created on 2014年9月17日
*/
public class CyclicBarrierTest {
public static void main(String[] args) throws InterruptedException {
final long start = System.currentTimeMillis();
final CyclicBarrier barrier = new CyclicBarrier(3, new Runnable() {
@Override
public void run() {
long end = System.currentTimeMillis();
System.out.println("导入" + 3 + "条数据,至此总共用时:" + (end - start)
+ "毫秒");
}
}); for (int i = 0; i < 9; i++) {
final int threadID = i + 1;
new Thread(new Runnable() {
@Override
public void run() {
try {
TimeUnit.SECONDS.sleep(new Random().nextInt(10));// 模拟业务操作
System.out.println(threadID + "完毕导入操作。 ");
barrier.await();
} catch (InterruptedException e) {
e.printStackTrace();
} catch (BrokenBarrierException e) {
e.printStackTrace();
}
}
}).start();
}
System.out.println("====主线程结束====");
}
}
运行结果为:
====主线程结束====
4完毕导入操作。 2完毕导入操作。
1完毕导入操作。
导入3条数据,至此总共用时:4006毫秒
5完毕导入操作。
6完毕导入操作。
8完毕导入操作。
导入3条数据。至此总共用时:4007毫秒
3完毕导入操作。 0完毕导入操作。
7完毕导入操作。
导入3条数据。至此总共用时:8006毫秒
程序没导入3条会进行一次计时,统计已经运行的时间。
假设CyclicBarrier构造函数的数字和for循环的次数相等的话,这个就是总共用时。
扩展
考虑一下上面的样例,假设for循环的次数不是CyclicBarrier监听次数的整数倍,比方是10。那运行结果将会是:
====主线程结束====
2完毕导入操作。
5完毕导入操作。 4完毕导入操作。 导入3条数据,至此总共用时:4005毫秒
8完毕导入操作。
1完毕导入操作。 3完毕导入操作。
导入3条数据。至此总共用时:5005毫秒
7完毕导入操作。 6完毕导入操作。 0完毕导入操作。 导入3条数据。至此总共用时:8005毫秒
9完毕导入操作。
在打印完“9完毕导入操作。
”之后,将一直等待。
在这里能够通过barrier.getNumberWaiting()查看还差多少个线程达到屏障点。
假设出现这样的情况。那就须要和CountDownLatch配合使用了。当子线程所有运行完。有推断barrier.getNumberWaiting()不等于0,则调用barrier.reset()重置。这个时候将会触发BrokenBarrierException异常,可是将结束整个过程。
改动的代码例如以下:
package howe.demo.thread.cyclicbarrier; import java.util.Random;
import java.util.concurrent.BrokenBarrierException;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.CyclicBarrier;
import java.util.concurrent.TimeUnit; /**
* @author liuxinghao
* @version 1.0 Created on 2014年9月17日
*/
public class CyclicBarrierTest {
public static void main(String[] args) throws InterruptedException {
final long start = System.currentTimeMillis();
final CountDownLatch count = new CountDownLatch(10);
final CyclicBarrier barrier = new CyclicBarrier(3, new Runnable() {
@Override
public void run() {
long end = System.currentTimeMillis();
System.out.println("导入" + 3 + "条数据,至此总共用时:" + (end - start)
+ "毫秒");
}
}); for (int i = 0; i < 10; i++) {
final int threadID = i + 1;
new Thread(new Runnable() {
@Override
public void run() {
try {
TimeUnit.SECONDS.sleep(new Random().nextInt(10));// 模拟业务操作
System.out.println(threadID + "完毕导入操作。 ");
count.countDown();
barrier.await();
} catch (InterruptedException e) {
e.printStackTrace();
} catch (BrokenBarrierException e) {
System.out.println("触发BrokenBarrierException异常。");
}
}
}).start();
}
count.await(); if(barrier.getNumberWaiting() != 0) {
System.out.println("不是整数倍。 都已运行完,重置CyclicBarrier。 ");
barrier.reset();
} System.out.println("====主线程结束====");
}
}
运行结果为:
3完毕导入操作。
9完毕导入操作。 6完毕导入操作。
导入3条数据,至此总共用时:3005毫秒
8完毕导入操作。
5完毕导入操作。 10完毕导入操作。
导入3条数据。至此总共用时:7005毫秒
1完毕导入操作。
7完毕导入操作。
4完毕导入操作。
2完毕导入操作。
导入3条数据,至此总共用时:9005毫秒
不是整数倍。都已运行完,重置CyclicBarrier。
====主线程结束====
触发BrokenBarrierException异常。
使用barrier.reset()进行重置,由于CyclicBarrier是一个循环,开头就是结尾,所以重置也能够理解为直接完毕。
另外。由于使用了CountDownLatch。所以主线程会锁定,直到线程通过count.await()向下运行。
多线程003 - 再谈CyclicBarrier的更多相关文章
- 沉淀再出发:再谈java的多线程机制
沉淀再出发:再谈java的多线程机制 一.前言 自从我们学习了操作系统之后,对于其中的线程和进程就有了非常深刻的理解,但是,我们可能在C,C++语言之中尝试过这些机制,并且做过相应的实验,但是对于ja ...
- 再谈多线程模型之生产者消费者(总结)(c++11实现)
0.关于 为缩短篇幅,本系列记录如下: 再谈多线程模型之生产者消费者(基础概念)(c++11实现) 再谈多线程模型之生产者消费者(单一生产者和单一消费者)(c++11实现) 再谈多线程模型之生产者消费 ...
- 再谈多线程模型之生产者消费者(多生产者和多消费者 )(c++11实现)
0.关于 为缩短篇幅,本系列记录如下: 再谈多线程模型之生产者消费者(基础概念)(c++11实现) 再谈多线程模型之生产者消费者(单一生产者和单一消费者)(c++11实现) 再谈多线程模型之生产者消费 ...
- 再谈多线程模型之生产者消费者(多生产者和单一消费者 )(c++11实现)
0.关于 为缩短篇幅,本系列记录如下: 再谈多线程模型之生产者消费者(基础概念)(c++11实现) 再谈多线程模型之生产者消费者(单一生产者和单一消费者)(c++11实现) 再谈多线程模型之生产者消费 ...
- 再谈多线程模型之生产者消费者(单一生产者和多消费者 )(c++11实现)
0.关于 为缩短篇幅,本系列记录如下: 再谈多线程模型之生产者消费者(基础概念)(c++11实现) 再谈多线程模型之生产者消费者(单一生产者和单一消费者)(c++11实现) 再谈多线程模型之生产者消费 ...
- 再谈多线程模型之生产者消费者(单一生产者和单一消费者)(c++11实现)
0.关于 为缩短篇幅,本系列记录如下: 再谈多线程模型之生产者消费者(基础概念)(c++11实现) 再谈多线程模型之生产者消费者(单一生产者和单一消费者)(c++11实现)[本文] 再谈多线程模型之生 ...
- 再谈多线程模型之生产者消费者(基础概念)(c++11实现)
0.关于 为缩短篇幅,本系列记录如下: 再谈多线程模型之生产者消费者(基础概念)(c++11实现)[本文] 再谈多线程模型之生产者消费者(单一生产者和单一消费者)(c++11实现) 再谈多线程模型之生 ...
- Another Look at Events(再谈Events)
转载:http://www.qtcn.org/bbs/simple/?t31383.html Another Look at Events(再谈Events) 最近在学习Qt事件处理的时候发现一篇很不 ...
- 再谈DOMContentLoaded与渲染阻塞—分析html页面事件与资源加载
浏览器的多线程中,有的线程负责加载资源,有的线程负责执行脚本,有的线程负责渲染界面,有的线程负责轮询.监听用户事件. 这些线程,根据浏览器自身特点以及web标准等等,有的会被浏览器特意的阻塞.两个很明 ...
随机推荐
- Linux下I/O复用 Select与Poll
Select #include <sys/time.h>#include <sys/types.h>#include <sys/unistd.h> int sele ...
- 自己定义Android Dialog
private void showDialog() { mDialog = new Dialog(this); mDialog.setCanceledOnTouchOutside(true); Win ...
- css sprite的实现
css sprite 为什么使用css sprite? 网页上的非常多静态小图片在载入时须要大量http请求,添加了响应时间.(哈哈.雅虎34条优化法则的第一条啊) css的background-po ...
- NSAttributedString编程
- (void)viewDidLoad { [super viewDidLoad]; NSMutableAttributedString *attributedString ...
- 捣鼓TinyMCE 粘贴图片并上传+Django后台
前面一篇写了上传到Flask后台,但是我不熟悉Flask,原先想学习一下,据说是轻量级. 但是我发现,学习会浪费我大量的时间,因为我并不是以这个为生的,我的目标只是要完成功能,让我自己能尽早使用起来, ...
- eclipse搭建android开发环境
1.首先安装JDK 此步骤是做JAVA必经之路,不多累述,强调要注意的地方: 目前为止android的开发环境只支持JDK1.7,千万不要下载JDK1.8. 下载的JDK一定要选择好操作系统,特别是要 ...
- Kylin基础教程(一)
一.Kylin介绍 1.1 现状 Hadoop于2006年初步实现,改变了企业级的大数据存储(基于HDFS)和批处理(主要基于MR)问题,10几年过去了,数据量随着互联网的发展井喷式增长,如何高速.低 ...
- python爬虫:爬取读者某一期内容
学会了怎么使用os模块 #!/usr/bin/python# -*- encoding:utf-8 -*- import requestsimport osfrom bs4 import Beauti ...
- Unity 退出游戏 方法
Application.Quit(); 嗯,没错,这篇就这么短.
- fullcalendar日历插件
https://www.helloweba.net/javascript/231.html