Java线程之Phaser
Phaser是一个灵活的线程同步工具,他包含了CyclicBarrier和CountDownLatch的相关功能
首先,来看一下如何用Phaser替代CountDownLatch。对于CountDownLatch而言,有2个重要的方法,一个是await()方法,可以使线程进入等待状态,在Phaser中,与之对应的方法是awaitAdvance(int n)。CountDownLatch中另一个重要的方法是countDown(),使计数器减一,当计数器为0时所有等待的线程开始执行,在Phaser中,与之对应的方法是arrive()。下面的例子创建了3个线程,打印一些字母,但是线程创建好后并不立刻执行,而是在主程序中对其进行控制,3秒钟后所有进程同时开始执行,一下是使用Phaser实现的版本,在注释中解释了如何改造成CountDownLatch版本。
- public class MyTest {
- public static void main(String[] args) {
- Phaser phaser = new Phaser(1); //此处可使用CountDownLatch(1)
- for(int i=0; i<3; i++) {
- new MyThread((char)(97+i), phaser).start();
- }
- try {
- TimeUnit.SECONDS.sleep(3);
- } catch (InterruptedException e) {
- e.printStackTrace();
- }
- phaser.arrive(); //此处可使用latch.countDown()
- }
- }
- class MyThread extends Thread {
- private char c;
- private Phaser phaser;
- public MyThread(char c, Phaser phaser) {
- this.c = c;
- this.phaser = phaser;
- }
- @Override
- public void run() {
- phaser.awaitAdvance(phaser.getPhase()); //此处可使用latch.await()
- for(int i=0; i<100; i++) {
- System.out.print(c+" ");
- if(i % 10 == 9) {
- System.out.println();
- }
- }
- }
- }
用Phaser替代CyclicBarrier更简单,CyclicBarrier的await()方法可以直接用Phaser的arriveAndAwaitAdvance()方法替代。
下面说说Phaser的高级用法,在Phaser内有2个重要状态,分别是phase和party。phase就是阶段,初值为0,当所有的线程执行完本轮任务,同时开始下一轮任务时,意味着当前阶段已结束,进入到下一阶段,phase的值自动加1。party就是线程,party=4就意味着Phaser对象当前管理着4个线程。Phaser还有一个重要的方法经常需要被重载,那就是boolean onAdvance(int phase, int registeredParties)方法。此方法有2个作用:1、当每一个阶段执行完毕,此方法会被自动调用,因此,重载此方法写入的代码会在每个阶段执行完毕时执行,相当于CyclicBarrier的barrierAction。2、当此方法返回true时,意味着Phaser被终止,因此可以巧妙的设置此方法的返回值来终止所有线程。例如:若此方法返回值为 phase>=3,其含义为当整个线程执行了4个阶段后,程序终止。
下面写个例子看看实际的应用。本例要实现的功能为:开启3个线程,分别打印字母a,b,c各10次,然后进入下一阶段打印后面的字母d,e,f各10次,然后再进入下一阶段.......以此类推,直到整个字母表全部打印完毕。在此期间主程序进入等待状态,直到3个工作线程全都结束,主程序才能结束。程序执行结果如下图所示:
代码如下:
- public class MyTest {
- public static void main(String[] args) {
- Phaser phaser = new Phaser(3) {//共有3个工作线程,因此在构造函数中赋值为3
- @Override
- protected boolean onAdvance(int phase, int registeredParties) {
- System.out.println("\n=========华丽的分割线=============");
- //本例中,当只剩一个线程时,这个线程必定是主线程,返回true表示终结
- return registeredParties == 1;
- }
- };
- System.out.println("程序开始执行");
- for(int i=0; i<3; i++) { //创建并启动3个线程
- new MyThread((char)(97+i), phaser).start();
- }
- phaser.register(); //将主线程动态增加到phaser中,此句执行后phaser共管理4个线程
- while(!phaser.isTerminated()) {//只要phaser不终结,主线程就循环等待
- int n = phaser.arriveAndAwaitAdvance();
- }
- //跳出上面循环后,意味着phaser终结,即3个工作线程已经结束
- System.out.println("程序结束");
- }
- }
- class MyThread extends Thread {
- private char c;
- private Phaser phaser;
- public MyThread(char c, Phaser phaser) {
- this.c = c;
- this.phaser = phaser;
- }
- @Override
- public void run() {
- while(!phaser.isTerminated()) {
- for(int i=0; i<10; i++) { //将当前字母打印10次
- System.out.print(c + " ");
- }
- //打印完当前字母后,将其更新为其后第三个字母,例如b更新为e,用于下一阶段打印
- c = (char) (c+3);
- if(c>'z') {
- //如果超出了字母z,则在phaser中动态减少一个线程,并退出循环结束本线程
- //当3个工作线程都执行此语句后,phaser中就只剩一个主线程了
- phaser.arriveAndDeregister();
- break;
- }else {
- //反之,等待其他线程到达阶段终点,再一起进入下一个阶段
- phaser.arriveAndAwaitAdvance();
- }
- }
- }
- }
最后,这篇文章写得非常好,看后收获非常大。http://whitesock.iteye.com/blog/1135457
Java线程之Phaser的更多相关文章
- Java线程之 InterruptedException 异常
Java线程之 InterruptedException 异常 当一个方法后面声明可能会抛出InterruptedException 异常时,说明该方法是可能会花一点时间,但是可以取消的方法. 抛 ...
- java 线程之executors线程池
一.线程池的作用 平时的业务中,如果要使用多线程,那么我们会在业务开始前创建线程,业务结束后,销毁线程.但是对于业务来说,线程的创建和销毁是与业务本身无关的,只关心线程所执行的任务.因此希望把尽可能多 ...
- java 线程之concurrent中的常用工具 CyclicBarrier
一.CyclicBarrier CyclicBarrier是一个同步辅助类,它允许一组线程互相等待,直到到达某个公共屏障点 (common barrier point).在涉及一组固定大小的线程的程序 ...
- C++/Java线程之分
JAVA线程状态图 1.C++/windows中主线程结束,其他线程必然死亡(即使调用pthread_detach解除父子关系,主线程消亡时也会导致子线程被迫关闭). ----1.1 一个进程中可以有 ...
- Java线程之CompletionService批处理任务
如果你向Executor提交了一个批处理任务,并且希望在它们完成后获得结果,怎么办呢? 为此你可以保存与每个任务相关联的Future,然后不断地调用 timeout为零的get,来检验Future是否 ...
- Java线程之Synchronized用法
synchronized是Java中的关键字,是一种同步锁.它修饰的对象有以下几种: 修饰一个代码块,被修饰的代码块称为同步语句块,其作用的范围是大括号{}括起来的代码,作用的对象是调用这个代码块的对 ...
- Java线程之Java内存模型(jmm)
一.Java内存模型(jmm) 线程通信 消息传递 重排序 顺序一致性 Happens-Before As-If-Serial
- Java线程之Dump
什么是线程dump Java Thread dump记录了线程在jvm中的执行信息,可以看成是线程活动的日志.Java线程转储文件有助于分析应用程序和死锁情况中的瓶颈. 如何获取线程转储文件 在这里, ...
- Java线程之FutureTask
简述 FutureTask是Future接口的实现类,并提供了可取消的异步处理的功能,它包含了启动和取消(start and cancel)任务的方法,同时也包含了可以返回FutureTask状态(c ...
随机推荐
- spl_autoload_register array参数
spl_autoload_register (PHP 5 >= 5.1.2) spl_autoload_register — 注册给定的函数作为 __autoload 的实现 说明¶ bool ...
- Flexbox完整指南- A Complete Guide to Flexbox
背景 Flexbox 布局 (FLexible Box)模块(现在处于W3C的最终征求意见稿(Last Call Working Draft)阶段)意在提供一个更为有效的方式来进行布局.对齐和分配一个 ...
- 关于修改动态库DLL的问题
最近在做一个小软件,需要做到新建.修改动态链接库(DLL)的功能.新建很简单,但是在修改的时候我突然发现问题来了.首先软件会在启动的时候将所有协议加载出来,然后做展示.此时你可以添加或减少动态库中的属 ...
- win7 64 安装Oracle 11G 、使用PLSQL进行连接 标准实践
第一步: 安装oracle 服务,两个解压包,分别解压后 合并到一个文件夹,点击exe安装 (安装过程中如遇到PATH问题,直接忽略即可) 第二步:使用SQLPlus 测试是否成功 安装成功:CMD ...
- Android开发第2篇 - Git插件安装
一.Eclipse上安装GIT插件EGit Eclipse的版本eclipse-java-helios-SR2-win32.zip(在Eclipse3.3版本找不到对应的 EGit插件,无法安装) E ...
- 用 BPL 封装数据连接
BPL 代码: uDM.pas unit uDM; interface uses SysUtils, Classes, uIntf, DB, ABSMain; type TDM = class(TDa ...
- WPF学习笔记3——Layout之1
一.概述 了解XAML的基本之后,进入Layout的学习.Layout,即布局,可能需要用到几种不同的容器.每一种容器都有各自的逻辑.在用户界面的设计过程中,很多时候是在想办法使得界面更加吸引.实在. ...
- UILocalNotification本地通知
// 执行通知一定要退出应用或挂起应用(进入后台)才能收到通知. 1.在iOS8及其以后版本中使用本地消息需要先获得用户的许可,否则无法成功注册本地消息.因此,我们将询问用户许可的代码片段添加到了ap ...
- Mac操作系统常用快捷键
复制:cmd+c 粘贴:cmd+v 剪切:先cmd+c,再cmd+opt+v 显示桌面:cmd+F3 切换输入法:cmd+space 打开Spotlight:ctrl+s ...
- 开发设计模式(五)单例模式(Singleton Pattern)
http://blog.sina.com.cn/s/blog_89d90b7c0101805m.html 单例模式:意思就是只有一个实例.单例模式确保某一个类只有一个实例,而且自行实例化并向整个系统提 ...