Java并发编程-移相器
移相器(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个阶段后,程序终止。
Phaser把多个线程执行的任务划分成多个阶段(phase),编程时要明确各个阶段的任务,每个阶段都可以有任意个参与者,线程可以随时注册并参与到某个阶段,当一个阶段中所有线程都成功完成之后,Phaser的onAdvance()被调用,可以通过覆盖添加自定义处理逻辑(类似循环屏障(关卡)使用的Runnable接口),然后Phaser类会自动进入下个阶段。如此循环,直到Phaser不再包含任何参与者。
register(),bulkRegister(),动态添加一个或多个参与者。
arrive(),某个参与者完成任务后调用
arriveAndDeregister(),任务完成,取消自己的注册。
arriveAndAwaitAdvance(),自己完成等待其他参与者完成:进入阻塞,直到Phaser成功进入下个阶段。
awaitAdvance()、awaitAdvanceInterruptibly(),等待phaser进入下个阶段,参数为当前阶段的编号,后者可以设置超时和处理中断请求。
另外,Phaser的一个重要特征是多个Phaser可以组成树形结构,Phaser提供了构造方法来指定当前对象的父对象;当一个子对象参与者>0,会自动注册到父对象中;当=0,自动解除注册。
package org.suxuan; import java.util.concurrent.Phaser;
import java.util.concurrent.atomic.AtomicReferenceArray; public class PhaserDemo {
public static void main(String[] args) {
final int workers = ;
final int workLength = ; final Phaser phaser = new Phaser(workers + );
final AtomicReferenceArray<String> lane1 = new AtomicReferenceArray<String>(new String[workLength]);
final AtomicReferenceArray<String> lane2 = new AtomicReferenceArray<String>(new String[workLength]); new Thread("Producer 1") {
@Override
public void run() {
for (int i = ; i < workLength; i++) {
$sleep(); lane1.set(i, "lane1-answer-" + i); System.out.printf("[%-17s] working in lane1 finished phase [%d]%n",
Thread.currentThread().getName(), phaser.getPhase()); phaser.arriveAndAwaitAdvance();
}
}
}.start(); new Thread("Slower producer 2") {
@Override
public void run() {
for (int i = ; i < workLength; i++) {
$sleep(); lane2.set(i, "lane2-answer-" + i); System.out.printf("[%-17s] working in lane2 finished phase [%d]%n",
Thread.currentThread().getName(), phaser.getPhase()); phaser.arriveAndAwaitAdvance();
}
}
}.start(); new Thread("Slow consumer") {
@Override
public void run() {
for (int start = ; start < workLength; ) {
System.out.printf("[%-17s] about to wait for phase [%d] completion%n",
Thread.currentThread().getName(), start); int phaseInProgress = phaser.awaitAdvance(start); //Read all the way up to the most recent completed phases.
for (int i = start; i < phaseInProgress; i++) {
System.out.printf("[%-17s] read [%s] & [%s] from phase [%d]%n",
Thread.currentThread().getName(), lane1.get(i), lane2.get(i), i);
} start = phaseInProgress; $sleep();
}
}
}.start(); phaser.arriveAndDeregister();
} private static void $sleep(long millis) {
try {
Thread.sleep(millis);
}
catch (InterruptedException e) {
}
}
}
Phaser初始化为3,之后customer线程取消自己的注册(此时只有两个生产者线程间进行同步),awaitAdvance不会阻塞,它直接返回。
两个生产者线程,依次生成结果放置到AtomicReferenceArray中。消费者线程每睡眠80Ms后,从结果集中,把当前已经完成的结果打印出来。
某次运行结果如下:
[Slow consumer ] about to wait for phase [] completion
[Producer ] working in lane1 finished phase []
[Slower producer ] working in lane2 finished phase []
[Slow consumer ] read [lane1-answer-] & [lane2-answer-] from phase []
[Producer ] working in lane1 finished phase []
[Slower producer ] working in lane2 finished phase []
[Producer ] working in lane1 finished phase []
[Slow consumer ] about to wait for phase [] completion
[Slow consumer ] read [lane1-answer-] & [lane2-answer-] from phase []
[Slower producer ] working in lane2 finished phase []
[Producer ] working in lane1 finished phase []
[Slower producer ] working in lane2 finished phase []
[Producer ] working in lane1 finished phase []
[Slower producer ] working in lane2 finished phase []
[Producer ] working in lane1 finished phase []
[Slow consumer ] about to wait for phase [] completion
[Slow consumer ] read [lane1-answer-] & [lane2-answer-] from phase []
[Slow consumer ] read [lane1-answer-] & [lane2-answer-] from phase []
[Slow consumer ] read [lane1-answer-] & [lane2-answer-] from phase []
[Slower producer ] working in lane2 finished phase []
[Producer ] working in lane1 finished phase []
[Slower producer ] working in lane2 finished phase []
[Slow consumer ] about to wait for phase [] completion
[Slow consumer ] read [lane1-answer-] & [lane2-answer-] from phase []
[Slow consumer ] read [lane1-answer-] & [lane2-answer-] from phase []
[Slower producer ] working in lane2 finished phase []
[Producer ] working in lane1 finished phase []
[Producer ] working in lane1 finished phase []
[Slower producer ] working in lane2 finished phase []
[Producer ] working in lane1 finished phase []
[Slow consumer ] about to wait for phase [] completion
[Slow consumer ] read [lane1-answer-] & [lane2-answer-] from phase []
[Slow consumer ] read [lane1-answer-] & [lane2-answer-] from phase []
[Slower producer ] working in lane2 finished phase []
[Slow consumer ] about to wait for phase [] completion
[Slow consumer ] read [lane1-answer-] & [lane2-answer-] from phase []
Java并发编程-移相器的更多相关文章
- Java并发编程-总纲
Java 原生支持并发,基本的底层同步包括:synchronized,用来标示一个方法(普通,静态)或者一个块需要同步执行(某一时刻,只允许一个线程在执行代码块).volatile,用来标识一个变量是 ...
- 【Java并发编程实战】----- AQS(四):CLH同步队列
在[Java并发编程实战]-–"J.U.C":CLH队列锁提过,AQS里面的CLH队列是CLH同步锁的一种变形.其主要从两方面进行了改造:节点的结构与节点等待机制.在结构上引入了头 ...
- 【Java并发编程实战】----- AQS(三):阻塞、唤醒:LockSupport
在上篇博客([Java并发编程实战]----- AQS(二):获取锁.释放锁)中提到,当一个线程加入到CLH队列中时,如果不是头节点是需要判断该节点是否需要挂起:在释放锁后,需要唤醒该线程的继任节点 ...
- 【Java并发编程实战】----- AQS(二):获取锁、释放锁
上篇博客稍微介绍了一下AQS,下面我们来关注下AQS的所获取和锁释放. AQS锁获取 AQS包含如下几个方法: acquire(int arg):以独占模式获取对象,忽略中断. acquireInte ...
- 【Java并发编程实战】-----“J.U.C”:CLH队列锁
在前面介绍的几篇博客中总是提到CLH队列,在AQS中CLH队列是维护一组线程的严格按照FIFO的队列.他能够确保无饥饿,严格的先来先服务的公平性.下图是CLH队列节点的示意图: 在CLH队列的节点QN ...
- 【Java并发编程实战】-----“J.U.C”:CountDownlatch
上篇博文([Java并发编程实战]-----"J.U.C":CyclicBarrier)LZ介绍了CyclicBarrier.CyclicBarrier所描述的是"允许一 ...
- 【Java并发编程实战】-----“J.U.C”:CyclicBarrier
在上篇博客([Java并发编程实战]-----"J.U.C":Semaphore)中,LZ介绍了Semaphore,下面LZ介绍CyclicBarrier.在JDK API中是这么 ...
- 【Java并发编程实战】-----“J.U.C”:ReentrantReadWriteLock
ReentrantLock实现了标准的互斥操作,也就是说在某一时刻只有有一个线程持有锁.ReentrantLock采用这种独占的保守锁直接,在一定程度上减低了吞吐量.在这种情况下任何的"读/ ...
- Java并发编程:volatile关键字解析
Java并发编程:volatile关键字解析 volatile这个关键字可能很多朋友都听说过,或许也都用过.在Java 5之前,它是一个备受争议的关键字,因为在程序中使用它往往会导致出人意料的结果.在 ...
随机推荐
- python- 动态加载目录下所有的类
# 背景 自动化测试框架中model层下有很多类,用来操作mysql的,使用的时候需要把全部的类加载进来,需要使用到动态加载类 # 解决方法 使用pkgutil,内置的方法,常用的话有两个方法 ite ...
- asp.net core 使用identityServer4的密码模式来进行身份认证(2) 认证授权原理
前言:本文将会结合asp.net core 认证源码来分析起认证的原理与流程.asp.net core版本2.2 对于大部分使用asp.net core开发的人来说. 下面这几行代码应该很熟悉了. s ...
- AEAI Portal 权限体系说明
1.概述 在数通畅联的产品体系中,AEAI Portal毫无疑问的占据了很重要的地位,在这里我们将通过参考Portal样例,讲述一下AEAI Portal权限体系的控制方法.在Portal使用过程中, ...
- [JSOI2018]列队(主席树)
跟上次那道列队不一样,但都是九条可怜...(吉老师太强了) 在主席树上统计答案,因为值域只有 \(10^6\) 甚至不用离散化... \(Code\ Below:\) #include <bit ...
- (干货) Android实现ImageVIew多点触控及双击缩放
支持多点触控,放大自由移动,双击可以放大缩小.直接上代码: package com.cbt.view; import android.content.Context; import android.g ...
- 跟着刚哥学习Spring框架--Spring容器(二)
Spring容器 启动Spring容器(实例化容器) -- IOC容器读取Bean配置创建Bean实例之前,必须对它进行实例化(加载启动),这样才可以从容器中获取Bean的实例并使用. Bean是S ...
- 网页关闭(解决window.close在火狐下不兼容问题)
熟悉前端的都知道,火狐默认状态非window.open的页面window.close是无效的 网上有很多人说,在火狐的地址栏输入:about:config然后找到dom.allow_scripts_t ...
- ThinkCMF后台验证码不显示,无法登陆怎么办?
ThinkCMF5在本地部署之后,过一段时间可能会莫名其妙的出现后台验证码不显示的问题,不明就里.着急登陆后台的话,可以先禁用后台验证码,方法如下: 打开文件:/app/admin/controlle ...
- 让PETSc跑得再快一些
最近做了一个使用PETSc来求解线性方程组(Ax=b)的项目,把其中遇到的一些坑和解决方法记录下来.本文不介绍PETSc如何入门,而是给出一些能让PETSc运行得更快的编程细节.开始我只是简单地修改P ...
- 05-02 Java 一维数组、内存分配、数组操作
数组的定义 动态初始化 /* 数组:存储同一种数据类型的多个元素的容器. 定义格式: A:数据类型[] 数组名; B:数据类型 数组名[]; 举例: A:int[] a; 定义一个int类型的数组a变 ...