Java并发基础:Phaser全面解析!

Java并发基础:Phaser全面解析! - 程序员古德

内容概要

Phaser是Java中一个灵活的同步工具,其优点在于支持多阶段的任务拆分与同步,并且能够动态地注册与注销参与者,它提供了丰富的等待与推进机制,使得开发者能够更细粒度地控制线程的协调行为,实现复杂的并行任务处理,相比于其他同步工具,Phaser更加灵活且易于扩展,适用于多种并发场景。

核心概念

在Java中,Phaser是一个灵活的同步工具类,它允许多个线程在一个或多个屏障(barrier points)上进行协调,可以把Phaser想象成一个多线程聚会的组织者,它负责确保所有参与的线程都到达某个阶段后再一起进行下一步。

举一个实际生活中的场景:假设正在开发一个在线多人游戏,比如,团队解谜游戏,在这个游戏中,有几个玩家(线程)需要合作完成一系列任务来通关,每个任务都被划分为几个阶段,而每个阶段都需要所有玩家共同完成某些操作后才能进入下一阶段。

这个场景中,Phaser就可以发挥它的作用,可以把每个阶段看作是一个屏障点,每个玩家线程在完成自己当前阶段的任务后会向Phaser报告,然后等待其他玩家完成,一旦所有玩家都完成了当前阶段的任务,Phaser就会像一个响铃一样,通知所有玩家可以进入下一阶段了。

比如,在解谜游戏的关卡中,四个玩家需要分别找到四个不同的线索,并将这些线索组合起来才能打开通往下一关的大门,每个玩家在找到线索后,都会告知Phaser自己已经完成任务,Phaser会等待所有四个玩家都找到线索后,再通知他们可以将线索组合起来打开大门进入下一关了。

Phaser主要用于解决多个线程分阶段共同完成任务的同步问题,它可以确保一组线程在达到某个屏障点(phase)之前都保持同步,即所有线程都完成了某个阶段的任务后,才能一起进入下一个阶段,这种同步机制尤其适用于需要多个线程协作完成复杂任务的情况,比如在线多人游戏、分布式系统、并行计算等场景。

Phaser在内部维护了一个状态机,用来跟踪和管理每个线程的执行状态以及各个阶段的完成情况,当一个线程完成任务并达到屏障点时,会调用Phaser的相应方法来通知其他线程,然后等待其他线程一起进入下一阶段,在这个过程中,Phaser会管理线程的同步和协作,确保所有线程都能按照预定的顺序完成各自的任务。

此外,Phaser还提供了灵活的注册和注销线程的功能,可以动态地添加或删除参与同步的线程,它还支持中断和超时机制,可以在等待其他线程的过程中被中断或设置超时,增强了对多线程同步的灵活性。

官方文档:https://docx.iamqiang.com/jdk11/api/java.base/java/util/concurrent/Phaser.html

代码案例

下面是一个简单的Java示例代码,演示了如何使用Phaser来同步多个线程,以确保它们分阶段完成任务,如下代码:

import java.util.concurrent.Phaser;  public class PhaserExample {  public static void main(String[] args) throws InterruptedException {  // 创建一个Phaser实例,初始时注册3个线程(不包括主线程)  Phaser phaser = new Phaser(3);  // 创建并启动3个线程  for (int i = 0; i < 3; i++) {  int threadNum = i + 1; // 为了在输出中区分线程  new Thread(() -> {  System.out.println("线程" + threadNum + ":已经准备好,等待其他线程。");  // 线程在此等待,直到所有线程都到达这个屏障点  phaser.arriveAndAwaitAdvance();  System.out.println("线程" + threadNum + ":第一阶段任务完成。");  // 模拟第二阶段的任务  try {  Thread.sleep(1000);  } catch (InterruptedException e) {  Thread.currentThread().interrupt();  return;  }  // 再次到达屏障点,等待其他线程  phaser.arriveAndAwaitAdvance();  System.out.println("线程" + threadNum + ":第二阶段任务完成。");  // 模拟第三阶段的任务  try {  Thread.sleep(1000);  } catch (InterruptedException e) {  Thread.currentThread().interrupt();  return;  }  // 最后一次到达屏障点,所有线程都完成后Phaser将自动进入终止状态  phaser.arriveAndAwaitAdvance();  System.out.println("线程" + threadNum + ":第三阶段任务完成,Phaser任务结束。");  }).start();  }  // 等待所有线程完成任务  // 注意:在实际应用中,可能不希望主线程在这里阻塞,而是去做其他工作  // 但为了演示目的,让主线程等待所有工作线程完成  phaser.awaitAdvance(phaser.getPhase());  System.out.println("所有线程的第一阶段任务完成。");  phaser.awaitAdvance(phaser.getPhase() + 1);  System.out.println("所有线程的第二阶段任务完成。");  phaser.awaitAdvance(phaser.getPhase() + 1);  System.out.println("所有线程的第三阶段任务完成,整个任务结束。");  }  
}

在上面代码中,创建了一个Phaser实例并初始注册了3个线程以及主线程,每个线程都执行三个阶段的任务,每个阶段任务之间都通过phaser.arriveAndAwaitAdvance()方法进行同步,每个线程在完成当前阶段的任务后,都会在这个方法上阻塞,直到所有其他线程也完成了它们当前阶段的任务,在所有线程都完成最后一个阶段的任务后,Phaser会自动进入终止状态,此时不会再有线程被阻塞。

上述代码输出如下结果:

主线程也已经准备好,等待其他线程。  
线程x已经准备好,等待其他线程。  
线程y已经准备好,等待其他线程。  
线程z已经准备好,等待其他线程。  
(这里所有线程和主线程都会等待,直到所有参与者都调用了arriveAndAwaitAdvance)  
主线程第一阶段任务完成(实际上主线程可能只是监控或协调其他线程)。  
线程x第一阶段任务完成。  
线程y第一阶段任务完成。  
线程z第一阶段任务完成。  
(所有线程和主线程继续执行,直到它们再次调用arriveAndAwaitAdvance)  
主线程第二阶段任务完成(实际上可能是等待其他线程完成某些任务)。  
线程x第二阶段任务完成。  
线程y第二阶段任务完成。  
线程z第二阶段任务完成。  
(所有线程和主线程继续执行第三阶段任务)  
主线程第三阶段任务完成(实际上可能是进行一些清理工作或者汇总结果)。  
线程x第三阶段任务完成,Phaser任务结束。  
线程y第三阶段任务完成,Phaser任务结束。  
线程z第三阶段任务完成,Phaser任务结束。

核心API

Phaser它允许一组线程互相等待,直到所有线程都到达某个屏障(barrier)点,Phaser非常适合用于多阶段的任务拆分和同步,以下是Phaser中一些重要方法的简要说明:

  1. Phaser(int parties)
    构造函数,创建一个新的Phaser实例,并设置注册的线程数(parties),这个数字表示在继续到下一个阶段之前,必须到达屏障的线程数。
  2. Phaser()
    构造函数,创建一个新的Phaser实例,但不设置注册的线程数,这通常用于层次结构的Phaser,其中子Phaser会继承父Phaser的注册线程数。
  3. register()
    增加一个到达屏障所需的线程数,如果调用此方法的线程尚未注册,它也会将自己注册为未到达的线程。
  4. arrive()
    表示当前线程已经到达屏障,并减少未到达的线程数,如果这是最后一个到达的线程,并且已经设置了下一个阶段的屏障,那么这个方法将返回true,否则返回false
  5. arriveAndAwaitAdvance()
    当前线程到达屏障,并等待其他线程也到达,当所有线程都到达后,屏障会自动推进到下一个阶段,然后该方法返回,如果当前Phaser被终止,这个方法会抛出IllegalStateException
  6. awaitAdvance(int phase)
    等待直到屏障推进到给定的阶段,如果当前阶段大于或等于给定的阶段,那么此方法将立即返回。
  7. isTerminated()
    检查Phaser是否已经终止,当注册的线程数减少到零,且没有新的线程注册时,Phaser将被终止。
  8. getPhase()
    获取当前屏障的阶段号,每个屏障都有一个唯一的阶段号,初始阶段号为0。
  9. getRegisteredParties()
    获取当前注册的线程数。
  10. getArrivedParties()
    获取已经到达当前屏障的线程数。
  11. getUnarrivedParties()
    获取尚未到达当前屏障的线程数,这实际上是getRegisteredParties()getArrivedParties()之间的差值。
  12. forceTermination()
    强制终止Phaser,即使还有未到达的线程,这会导致所有等待在arriveAndAwaitAdvance()awaitAdvance(int)方法上的线程抛出IllegalStateException
  13. onAdvance(int phase, int registeredParties)
    这是一个受保护的方法,可以在子类中覆盖,以便在每个屏障阶段推进时执行自定义操作。
  14. bulkRegister(int parties)
    一次性注册多个线程,这通常用于静态已知的线程数,或者当多个任务由同一个线程代表时。

核心总结

Java并发基础:Phaser全面解析! - 程序员古德

Phaser类的目的是允许在并发编程中同步多个线程之间的执行,它具有如下优点,如下:

  1. 更好的可扩展性:Phaser类相对于其他同步工具类(如CyclicBarrier和CountDownLatch)具有更好的可扩展性,因为它支持更多的参与者(即线程)同时进行同步。
  2. 自动注销和清理:当所有参与者都完成执行后,Phaser会自动注销并释放相关资源,这有助于避免内存泄漏和资源浪费。
  3. 灵活的执行模式:Phaser类提供了多种执行模式,如并行、串行和混合模式,这使得在处理并发任务时更加灵活。

它也有不少缺点,如:1、与其他的同步工具类相比,Phaser类的实现相对复杂,因此在某些场景下可能会引入额外的性能开销,并且Phaser类具有一定的使用门槛,使用时深入理解并发编程和Java并发API,这可能会增加学习成本。

关注我,每天学习互联网编程技术 - 程序员古德

END!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/669009.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Redis核心技术与实战【学习笔记】 - 25.Redis 支撑秒杀场景的关键技术

简述 秒杀是一个非常经典的活动场景&#xff0c;比如&#xff0c;在双 11、618 等电商促销活动中&#xff0c;都会有秒杀场景。秒杀场景的业务特点是限时限量&#xff0c;业务系统要处理瞬时的大量高并发请求&#xff0c;而 Redis 就经常被用来支撑秒杀活动。 秒杀场景包含多…

2.4日总结

第一题&#xff1a;选数 题解&#xff1a;思路还是很简单的&#xff0c;只需要想清楚dfs里的函数都是什么就可以了&#xff0c;还有一个简单的判断素数的函数&#xff0c;这题真没啥难度&#xff0c;就是属于基础题吧&#xff0c;请看AC代码 #include <stdio.h> #includ…

【c/python】GtkGrid

一、GtkGrid GtkGrid 是 GTK (GIMP Toolkit) 中的一个基础容器构件&#xff08;widget&#xff09;&#xff0c;它可以用来安排其他构件在一个灵活的多行多列的网格中。每个加入网格的构件都可以占据一个或多个行和列。由于 GtkGrid 提供了在二维空间中安排构件的方式&#xf…

YOLOv5独家改进:上采样算子 | 超轻量高效动态上采样DySample,效果秒杀CAFFE,助力小目标检测

💡💡💡本文独家改进:一种超轻量高效动态上采样DySample, 具有更少的参数、FLOPs,效果秒杀CAFFE和YOLOv5网络中的nn.Upsample 💡💡💡在多个数据集下验证能够涨点,尤其在小目标检测领域涨点显著。 收录 YOLOv5原创自研 https://blog.csdn.net/m0_63774211/cate…

多输入多输出 | Matlab实现PSO-LSTM粒子群优化长短期记忆神经网络多输入多输出预测

多输入多输出 | Matlab实现PSO-LSTM粒子群优化长短期记忆神经网络多输入多输出预测 目录 多输入多输出 | Matlab实现PSO-LSTM粒子群优化长短期记忆神经网络多输入多输出预测预测效果基本介绍程序设计参考资料 预测效果 基本介绍 Matlab实现PSO-LSTM粒子群优化长短期记忆神经网络…

2 物理层(一):数据通信的基本概念

目录 目标1 数据通信的基本概念1.1 数据通信的基本概念1、数据2、信息3、信号4、信道5、通信和数据通信6、数据通信网7、码元和码字8、数据分组9、基带传输、频带传输和宽带传输基带传输频带传输宽带传输 1.2 数据通信的主要技术指标1.3 数据通信系统1、数据通信系统基本模型编…

【ARM 嵌入式 编译系列 2.7 -- GCC 编译优化参数详细介绍】

文章目录 GCC 编译优化概述常用优化等级-O1 打开的优化选项-O2 打开的优化选项-O3 打开的优化选项-Os 打开的优化选项优化技术使用优化选项的注意事项GCC 编译优化概述 GCC(GNU Compiler Collection)包含了用于C、C++、Objective-C、Fortran、Ada和Go等语言的编译器。在编译…

能和ai聊天的软件有吗?分享4款智能软件!

随着科技的飞速发展&#xff0c;人工智能&#xff08;AI&#xff09;已经渗透到我们生活的方方面面。如今&#xff0c;我们只需一款软件&#xff0c;就能与AI进行流畅的对话。今天&#xff0c;就让我们一起揭开这些神秘软件的神秘面纱&#xff0c;看看它们如何让我们的沟通变得…

ONLYOFFICE:一站式办公,探索高效办公新境界

写在前面ONLYOFFICE 介绍ONLYOFFICE 有哪些优势ONLYOFFICE 文档 8.0 发布如何体验 ONLYOFFICEONLYOFFICE 文档部分页面截图 写在前面 在当今这样一个数字化时代&#xff0c;办公软件已经成为我们日常工作中不可或缺的一部分&#xff0c;熟练使用 Office、WPS、腾讯文档、金山文…

搜大学英语题,用哪个软件好?哪款大学搜题工具好用? #媒体#职场发展#媒体

大学生除了学习专业知识外&#xff0c;还应该关注和学习一些软技能&#xff0c;如沟通能力、团队合作和领导力等&#xff0c;以提升自己的综合素质。 1.好大学在线 好大学在线是上海交通大学拥有的中国顶尖慕课平台。 依托该平台&#xff0c;上海交通大学与百度及金智教育实…

机器学习 | 基于网格搜索的SVM超参数调节

机器学习模型被定义为一个数学模型&#xff0c;其中包含许多需要从数据中学习的参数。然而&#xff0c;有一些参数&#xff0c;称为超参数&#xff0c;这些参数不能直接学习。它们通常是由人类在实际训练开始前根据直觉或经验和试验选择的。这些参数通过提高模型的性能&#xf…

网络协议梳理

1 引言 在计算机网络中要做到有条不紊地交换数据&#xff0c;就必须遵守一些事先约定好的规则。这些规则明确规定了所交换的数据的格式以及有关的同步问题。这里所说的同步不是狭义的&#xff08;即同频或同频同相&#xff09;而是广义的&#xff0c;即在一定的条件下应当发生什…

每日一题——LeetCode1403.非递增顺序的最小子序列

方法一 个人方法&#xff1a; 按题目要求&#xff0c;尽可能先取出nums里最大的值&#xff0c;这样才能满足子序列尽可能短且元素之和最大 var minSubsequence function(nums) {nums.sort((a,b)>a-b)let sum1nums.reduce((a,b)>ab,0),sum20,res[]while(sum1>sum2){…

【并发编程】原子累加器

&#x1f4dd;个人主页&#xff1a;五敷有你 &#x1f525;系列专栏&#xff1a;并发编程 ⛺️稳重求进&#xff0c;晒太阳 JDK8之后有专门做累加的类&#xff0c;效率比自己做快数倍以上 累加器性能比较 参数是方法 // supplier 提供者 无中生有 ()->结果// func…

golang并发安全-sync.Once

什么是sync.Once sync.Once 是 Go 语言中的一种同步原语&#xff0c;用于确保某个操作或函数在并发环境下只被执行一次。它只有一个导出的方法&#xff0c;即 Do&#xff0c;该方法接收一个函数参数。在 Do 方法被调用后&#xff0c;该函数将被执行&#xff0c;而且只会执行一…

Excel——高级筛选匹配条件提取数据

一、筛选多条件 Q&#xff1a;筛选多个条件&#xff0c;并将筛选出的内容复制到其他区域 点击任意一个单元格 点击【数据】——【筛选】——【高级筛选】 选择【将筛选结果复制到其他位置】——在【列表区域】 鼠标选择对应的区域位置&#xff0c;条件区域一定要单独写出来&a…

政安晨:示例演绎Python的函数与获取帮助的方法

调用函数和定义我们自己的函数&#xff0c;并使用Python内置的文档&#xff0c;是成为一位Pythoner的开始。 通过我的上篇文章&#xff0c;相信您已经看过并使用了print和abs等函数。但是Python还有许多其他函数&#xff0c;并且定义自己的函数是Python编程的重要部分。 在本…

【51单片机】LED的三个基本项目(LED点亮&LED闪烁&LED流水灯)(3)

前言 大家好吖&#xff0c;欢迎来到 YY 滴单片机系列 &#xff0c;热烈欢迎&#xff01; 本章主要内容面向接触过单片机的老铁 主要内容含&#xff1a; 欢迎订阅 YY滴C专栏&#xff01;更多干货持续更新&#xff01;以下是传送门&#xff01; YY的《C》专栏YY的《C11》专栏YY的…

将xyz格式的GRACE数据转成geotiff格式

我们需要将xyz格式的文件转成geotiff便于成图&#xff0c;或者geotiff转成xyz用于数据运算&#xff0c;下面介绍如何实现这一操作&#xff0c;采用GMT和matlab两种方法。 1.GMT转换 我们先准备一个xyz文件&#xff0c;这里是一个降水文件。在gmt中采用以下的语句实现xyz转grd…

java synchronized浅析

介绍synchronized synchronized 是Java编程语言中的一个关键字&#xff0c;用于实现线程间的同步。在多线程环境下&#xff0c;它确保了共享资源在同一时刻只能被一个线程访问或修改&#xff0c;从而避免了因多个线程并发操作同一数据而导致的数据不一致和竞态条件。 synchroni…