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的更多相关文章

  1. 沉淀再出发:再谈java的多线程机制

    沉淀再出发:再谈java的多线程机制 一.前言 自从我们学习了操作系统之后,对于其中的线程和进程就有了非常深刻的理解,但是,我们可能在C,C++语言之中尝试过这些机制,并且做过相应的实验,但是对于ja ...

  2. 再谈多线程模型之生产者消费者(总结)(c++11实现)

    0.关于 为缩短篇幅,本系列记录如下: 再谈多线程模型之生产者消费者(基础概念)(c++11实现) 再谈多线程模型之生产者消费者(单一生产者和单一消费者)(c++11实现) 再谈多线程模型之生产者消费 ...

  3. 再谈多线程模型之生产者消费者(多生产者和多消费者 )(c++11实现)

    0.关于 为缩短篇幅,本系列记录如下: 再谈多线程模型之生产者消费者(基础概念)(c++11实现) 再谈多线程模型之生产者消费者(单一生产者和单一消费者)(c++11实现) 再谈多线程模型之生产者消费 ...

  4. 再谈多线程模型之生产者消费者(多生产者和单一消费者 )(c++11实现)

    0.关于 为缩短篇幅,本系列记录如下: 再谈多线程模型之生产者消费者(基础概念)(c++11实现) 再谈多线程模型之生产者消费者(单一生产者和单一消费者)(c++11实现) 再谈多线程模型之生产者消费 ...

  5. 再谈多线程模型之生产者消费者(单一生产者和多消费者 )(c++11实现)

    0.关于 为缩短篇幅,本系列记录如下: 再谈多线程模型之生产者消费者(基础概念)(c++11实现) 再谈多线程模型之生产者消费者(单一生产者和单一消费者)(c++11实现) 再谈多线程模型之生产者消费 ...

  6. 再谈多线程模型之生产者消费者(单一生产者和单一消费者)(c++11实现)

    0.关于 为缩短篇幅,本系列记录如下: 再谈多线程模型之生产者消费者(基础概念)(c++11实现) 再谈多线程模型之生产者消费者(单一生产者和单一消费者)(c++11实现)[本文] 再谈多线程模型之生 ...

  7. 再谈多线程模型之生产者消费者(基础概念)(c++11实现)

    0.关于 为缩短篇幅,本系列记录如下: 再谈多线程模型之生产者消费者(基础概念)(c++11实现)[本文] 再谈多线程模型之生产者消费者(单一生产者和单一消费者)(c++11实现) 再谈多线程模型之生 ...

  8. Another Look at Events(再谈Events)

    转载:http://www.qtcn.org/bbs/simple/?t31383.html Another Look at Events(再谈Events) 最近在学习Qt事件处理的时候发现一篇很不 ...

  9. 再谈DOMContentLoaded与渲染阻塞—分析html页面事件与资源加载

    浏览器的多线程中,有的线程负责加载资源,有的线程负责执行脚本,有的线程负责渲染界面,有的线程负责轮询.监听用户事件. 这些线程,根据浏览器自身特点以及web标准等等,有的会被浏览器特意的阻塞.两个很明 ...

随机推荐

  1. java书籍推荐:《Java SE 6 技術手册》

    Java SE 6 技術手册 或  Java SE 6 技術手册 Java SE 6 技術手册 為什麼選擇用 Markdown?仅仅是單純把文件又一次排版太無聊了,不如趁這個機會學些新東西.所以我就藉 ...

  2. [Performance] Optimize Paint and Composite for the website

    "Paint" is one of the most preference killer, it can easily cost more than 60fps, and once ...

  3. [HTML5] Semantics for accessibility

    For example, when we use checkbox, if we do like this: <div class="inline-control sign-up co ...

  4. openssl之EVP系列之6---EVP_Encrypt系列函数编程架构及样例

    openssl之EVP系列之6---EVP_Encrypt系列函数编程架构及样例     ---依据openssl doc/crypto/EVP_EncryptInit.pod和doc/ssleay. ...

  5. C++ 浅析 STL 中的 list 容器

    list - 擅长插入删除的链表 链表对于数组来说就是相反的存在. 数组本身是没有动态增长能力的(程序中也必须又一次开辟内存来实现), 而链表强悍的就是动态增长和删除的能力. 但对于数组强悍的随机訪问 ...

  6. MyEclipse2015安装SVN插件

    一.下载SVN插件subclipse 下载地址:http://subclipse.tigris.org/servlets/ProjectDocumentList?folderID=2240 在打开的网 ...

  7. Swift - 使用CollectionView实现图片Gallery画廊效果(左右滑动浏览图片)

    1,效果图 (1)图片从左至右横向排列(只有一行),通过手指拖动可以前后浏览图片. (2)视图滚动时,每张图片根据其与屏幕中心距离的不同,显示尺寸也会相应地变化.越靠近屏幕中心尺寸就越大,远离屏幕中心 ...

  8. 使用dbms_metadata.get_ddl遇到ORA-31603

    建了一个外部表,想看看这个表的信息,一查就报错了: SQL> select dbms_metadata.get_ddl('TABLE','ext_case1') from dual; ERROR ...

  9. Oracle数据库中闪回恢复的详细分析

    Oracle9i开始提供闪回查询,以便能在需要的时候查到过去某个时刻的一致性数据,这是通过Undo实现的.这个功能有很大的限制,就是相关事务的undo不能被覆盖,否则就无力回天了.oracle10g大 ...

  10. The Vertica Analytic Database:C-Store 7 Years Later笔记

    1.设计目标 Vertica数据库可以说是7年之后的C-Store,在2012年发表的这样一篇论文,描述了现在基于C-Store的一部分改进,当然,Vertica借鉴了很多C-Store的思想,但并非 ...