Phaser是一个灵活的线程同步工具,他包含了CyclicBarrier和CountDownLatch的相关功能

首先,来看一下如何用Phaser替代CountDownLatch。对于CountDownLatch而言,有2个重要的方法,一个是await()方法,可以使线程进入等待状态,在Phaser中,与之对应的方法是awaitAdvance(int n)。CountDownLatch中另一个重要的方法是countDown(),使计数器减一,当计数器为0时所有等待的线程开始执行,在Phaser中,与之对应的方法是arrive()。下面的例子创建了3个线程,打印一些字母,但是线程创建好后并不立刻执行,而是在主程序中对其进行控制,3秒钟后所有进程同时开始执行,一下是使用Phaser实现的版本,在注释中解释了如何改造成CountDownLatch版本。

  1. public class MyTest {
  2. public static void main(String[] args) {
  3. Phaser phaser = new Phaser(1); //此处可使用CountDownLatch(1)
  4. for(int i=0; i<3; i++) {
  5. new MyThread((char)(97+i), phaser).start();
  6. }
  7. try {
  8. TimeUnit.SECONDS.sleep(3);
  9. } catch (InterruptedException e) {
  10. e.printStackTrace();
  11. }
  12. phaser.arrive();  //此处可使用latch.countDown()
  13. }
  14. }
  15. class MyThread extends Thread {
  16. private char c;
  17. private Phaser phaser;
  18. public MyThread(char c, Phaser phaser) {
  19. this.c = c;
  20. this.phaser = phaser;
  21. }
  22. @Override
  23. public void run() {
  24. phaser.awaitAdvance(phaser.getPhase()); //此处可使用latch.await()
  25. for(int i=0; i<100; i++) {
  26. System.out.print(c+" ");
  27. if(i % 10 == 9) {
  28. System.out.println();
  29. }
  30. }
  31. }
  32. }

用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个工作线程全都结束,主程序才能结束。程序执行结果如下图所示:

代码如下:

  1. public class MyTest {
  2. public static void main(String[] args) {
  3. Phaser phaser = new Phaser(3) {//共有3个工作线程,因此在构造函数中赋值为3
  4. @Override
  5. protected boolean onAdvance(int phase, int registeredParties) {
  6. System.out.println("\n=========华丽的分割线=============");
  7. //本例中,当只剩一个线程时,这个线程必定是主线程,返回true表示终结
  8. return registeredParties == 1;
  9. }
  10. };
  11. System.out.println("程序开始执行");
  12. for(int i=0; i<3; i++) { //创建并启动3个线程
  13. new MyThread((char)(97+i), phaser).start();
  14. }
  15. phaser.register(); //将主线程动态增加到phaser中,此句执行后phaser共管理4个线程
  16. while(!phaser.isTerminated()) {//只要phaser不终结,主线程就循环等待
  17. int n = phaser.arriveAndAwaitAdvance();
  18. }
  19. //跳出上面循环后,意味着phaser终结,即3个工作线程已经结束
  20. System.out.println("程序结束");
  21. }
  22. }
  23. class MyThread extends Thread {
  24. private char c;
  25. private Phaser phaser;
  26. public MyThread(char c, Phaser phaser) {
  27. this.c = c;
  28. this.phaser = phaser;
  29. }
  30. @Override
  31. public void run() {
  32. while(!phaser.isTerminated()) {
  33. for(int i=0; i<10; i++) { //将当前字母打印10次
  34. System.out.print(c + " ");
  35. }
  36. //打印完当前字母后,将其更新为其后第三个字母,例如b更新为e,用于下一阶段打印
  37. c = (char) (c+3);
  38. if(c>'z') {
  39. //如果超出了字母z,则在phaser中动态减少一个线程,并退出循环结束本线程
  40. //当3个工作线程都执行此语句后,phaser中就只剩一个主线程了
  41. phaser.arriveAndDeregister();
  42. break;
  43. }else {
  44. //反之,等待其他线程到达阶段终点,再一起进入下一个阶段
  45. phaser.arriveAndAwaitAdvance();
  46. }
  47. }
  48. }
  49. }

最后,这篇文章写得非常好,看后收获非常大。http://whitesock.iteye.com/blog/1135457

Java线程之Phaser的更多相关文章

  1. Java线程之 InterruptedException 异常

    Java线程之 InterruptedException 异常   当一个方法后面声明可能会抛出InterruptedException 异常时,说明该方法是可能会花一点时间,但是可以取消的方法. 抛 ...

  2. java 线程之executors线程池

    一.线程池的作用 平时的业务中,如果要使用多线程,那么我们会在业务开始前创建线程,业务结束后,销毁线程.但是对于业务来说,线程的创建和销毁是与业务本身无关的,只关心线程所执行的任务.因此希望把尽可能多 ...

  3. java 线程之concurrent中的常用工具 CyclicBarrier

    一.CyclicBarrier CyclicBarrier是一个同步辅助类,它允许一组线程互相等待,直到到达某个公共屏障点 (common barrier point).在涉及一组固定大小的线程的程序 ...

  4. C++/Java线程之分

    JAVA线程状态图 1.C++/windows中主线程结束,其他线程必然死亡(即使调用pthread_detach解除父子关系,主线程消亡时也会导致子线程被迫关闭). ----1.1 一个进程中可以有 ...

  5. Java线程之CompletionService批处理任务

    如果你向Executor提交了一个批处理任务,并且希望在它们完成后获得结果,怎么办呢? 为此你可以保存与每个任务相关联的Future,然后不断地调用 timeout为零的get,来检验Future是否 ...

  6. Java线程之Synchronized用法

    synchronized是Java中的关键字,是一种同步锁.它修饰的对象有以下几种: 修饰一个代码块,被修饰的代码块称为同步语句块,其作用的范围是大括号{}括起来的代码,作用的对象是调用这个代码块的对 ...

  7. Java线程之Java内存模型(jmm)

    一.Java内存模型(jmm) 线程通信 消息传递 重排序 顺序一致性 Happens-Before As-If-Serial

  8. Java线程之Dump

    什么是线程dump Java Thread dump记录了线程在jvm中的执行信息,可以看成是线程活动的日志.Java线程转储文件有助于分析应用程序和死锁情况中的瓶颈. 如何获取线程转储文件 在这里, ...

  9. Java线程之FutureTask

    简述 FutureTask是Future接口的实现类,并提供了可取消的异步处理的功能,它包含了启动和取消(start and cancel)任务的方法,同时也包含了可以返回FutureTask状态(c ...

随机推荐

  1. 在JQuery和Js中,如何让ajax执行完后再继续往下执行 async

    async (默认: true) 默认设置下,所有请求均为异步请求.如果需要发送同步请求,请将此选项设置为 false.注意,同步请求将锁住浏览器,用户其它操作必须等待请求完成才可以执行. var t ...

  2. Android Studio API 文档_下载与使用

    如何下载API 说明: 时间: 2016/7/9 根据百度经验步骤改编(百度经验), 但是比它更好, 亲测可用 1.1 下载API文档: 1.1.1 SDK Manager 1.1.2 1.1.3 ( ...

  3. php class类的用法详细总结

    以下是对php中class类的用法进行了详细的总结介绍,需要的朋友可以过来参考下 一:结构和调用(实例化): class className{} ,调用:$obj = new className(); ...

  4. Java中的异常处理机制的简单原理和应用

    异常是指java程序运行时(非编译)所发生的非正常情况或错误,与现实生活中的事件很相似,现实生活中的事件可以包含事件发生的时间.地点.人物.情节等信息,可以用一个对象来表示,Java使用面向对象的方式 ...

  5. windows下使用MinGW的调试工具gdb.exe调试C程序

    1.编译源代码 C:MinGW\bin>gcc.exe -g -o program.exe program.c 编译选项上要加上“g”,这样生成的目标程序会含有调试内容,再用gdb调试的时候才能 ...

  6. SQL导入

    然后将新窗口中所有内容放到你需要复制的那个数据库中->新建查询->修改第一行 USE[新数据库名]-> 运行这段代码->刷新数据库 基本就是选择源数据库和目标数据库,特别注意的 ...

  7. C# Windows Service调用IBM Lotus Notes发送邮件

    近日研究了下IBM Lotus Mail,这货果然是麻烦,由于公司策略,没有开放smtp,很多系统邮件都没有办法发送,于是入手google学习Lotus Mail,想做成Windows服务,提供wcf ...

  8. 软件测试 -- 测试人员和QA的区别

    软件测试人员的职责是尽可能早的找出软件缺陷,确保得以修复. 而质量保证人员(QA)主要职责是创建或者制定标准和方法,提高促进软件开发能力和减少软件缺陷. 测试人员的主要工作是测试,质量保证人员日常工作 ...

  9. 解决WP8应用里ListBox绑定数据变多导致越来越卡

    ListBox控件绑定数据,当滑动到底部的时候加载数据到列表上,这样就会产生一个问题,当ListBox上面绑定的数据有几千条的时候,界面将会卡顿,我们可以通过在ListBox上只绑定指定数量的数据,其 ...

  10. js数值计算

    js在小数数值计算时会出现误差,比如0.19+15.02=15.20999999999999,出现此问题的原因,百度上有,为了避免误差产生可以这样做:(0.19*100+15.02*100)/100.