• Java线程之Phaser


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

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

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

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

    代码如下:

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

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

  • 相关阅读:
    .NET 中验证控件的使用
    Input(file) 控件的简单使用!
    dropdownlist 、listbox 与 panel的使用
    Applicatin、 server、 session 、cookies对象的简单使用方法
    三天晚上看了24集 央视版《神雕侠侣》 还不错
    推荐一本 asp.net(c#)学习的好书
    .net 中广告控件 AdRotator 的使用(xml)
    Asp.net 1.0 升级至 ASP.NET 2.0十个问题总结
    多线程操作 同一个textbox. yi
    博客开张第一贴!谢谢!大伙多多关照!谢谢. yi
  • 原文地址:https://www.cnblogs.com/duanxz/p/4372954.html
Copyright © 2020-2023  润新知