使用信号量解决并发问题

使用信号量解决并发问题

本文主要讨论的是java的信号量在并发过程中的应用。信号量Semaphore的具体实现如下:

package java.util.concurrent;
import java.util.Collection;
import java.util.concurrent.locks.AbstractQueuedSynchronizer;public class Semaphore implements java.io.Serializable {private static final long serialVersionUID = -3222578661600680210L;/** 同步机制基于AQS实现 **/private final Sync sync;/*** 信号量的同步器实现,使用AQS的状态作为许可,子类提供了公平和非公平版本。*/abstract static class Sync extends AbstractQueuedSynchronizer {private static final long serialVersionUID = 1192457210091910933L;Sync(int permits) {setState(permits);}final int getPermits() {return getState();}final int nonfairTryAcquireShared(int acquires) {for (;;) {int available = getState();int remaining = available - acquires;if (remaining < 0 ||compareAndSetState(available, remaining))return remaining;}}protected final boolean tryReleaseShared(int releases) {for (;;) {int current = getState();int next = current + releases;if (next < current) // overflowthrow new Error("Maximum permit count exceeded");if (compareAndSetState(current, next))return true;}}final void reducePermits(int reductions) {for (;;) {int current = getState();int next = current - reductions;if (next > current) // underflowthrow new Error("Permit count underflow");if (compareAndSetState(current, next))return;}}final int drainPermits() {for (;;) {int current = getState();if (current == 0 || compareAndSetState(current, 0))return current;}}}/*** 非公平版本*/static final class NonfairSync extends Sync {private static final long serialVersionUID = -2694183684443567898L;NonfairSync(int permits) {super(permits);}protected int tryAcquireShared(int acquires) {return nonfairTryAcquireShared(acquires);}}/*** 公平版本*/static final class FairSync extends Sync {private static final long serialVersionUID = 2014338818796000944L;FairSync(int permits) {super(permits);}protected int tryAcquireShared(int acquires) {for (;;) {if (hasQueuedPredecessors())return -1;int available = getState();int remaining = available - acquires;if (remaining < 0 ||compareAndSetState(available, remaining))return remaining;}}}/*** 使用给定数量的许可创建一个非公平的信号量对象*/public Semaphore(int permits) {sync = new NonfairSync(permits);}/*** 使用指定的许可创建一个可选公平或非公平版本的信号量对象*/public Semaphore(int permits, boolean fair) {sync = fair ? new FairSync(permits) : new NonfairSync(permits);}/**     * 从信号量对象中获取一个许可,会阻塞等待许可可用或者线程被中断,同时可用的许可数量减一。*/public void acquire() throws InterruptedException {sync.acquireSharedInterruptibly(1);}/*** 从信号量对象中获取一个许可,会阻塞等待许可可用,同时可用的许可数量减一。              */public void acquireUninterruptibly() {sync.acquireShared(1);}/*** 从信号量对象中获取一个许可,当且仅当调用时可用则立即返回true,否则直接给返回false。*/public boolean tryAcquire() {return sync.nonfairTryAcquireShared(1) >= 0;}/*** 从信号量对象中获取一个许可,如果许可在给定等待时间内变得可用并且当前线程未被中断。     */public boolean tryAcquire(long timeout, TimeUnit unit)throws InterruptedException {return sync.tryAcquireSharedNanos(1, unit.toNanos(timeout));}/*** 释放一个许可,返回给信号量。     */public void release() {sync.releaseShared(1);}/*** 从信号量中获取给定数量的许可,阻塞等待许可可用。*/public void acquire(int permits) throws InterruptedException {if (permits < 0) throw new IllegalArgumentException();sync.acquireSharedInterruptibly(permits);}/*** 从信号量中获取给定数量的许可,阻塞等待所有的许可可用。*/public void acquireUninterruptibly(int permits) {if (permits < 0) throw new IllegalArgumentException();sync.acquireShared(permits);}/*** 从信号量中获取给定数量的许可,当且仅当本次访问时所有许可可用。*/public boolean tryAcquire(int permits) {if (permits < 0) throw new IllegalArgumentException();return sync.nonfairTryAcquireShared(permits) >= 0;}/*** 从信号量中获取给定数量的许可,如果许可在给定等待时间内变得可用并且当前线程未被中断。  */public boolean tryAcquire(int permits, long timeout, TimeUnit unit)throws InterruptedException {if (permits < 0) throw new IllegalArgumentException();return sync.tryAcquireSharedNanos(permits, unit.toNanos(timeout));}/*** 释放给定数量的许可,返回信号量。    */public void release(int permits) {if (permits < 0) throw new IllegalArgumentException();sync.releaseShared(permits);}/*** 获取当前信号量中可用的许可*/public int availablePermits() {return sync.getPermits();}/*** 获取立即可用的许可*/public int drainPermits() {return sync.drainPermits();}/*** 通过指示的减少减少可用许可证的数量。*/protected void reducePermits(int reduction) {if (reduction < 0) throw new IllegalArgumentException();sync.reducePermits(reduction);}/*** 如果此信号量的公平性设置为true,则返回true*/public boolean isFair() {return sync instanceof FairSync;}/*** 查询是否有线程在等待获取*/public final boolean hasQueuedThreads() {return sync.hasQueuedThreads();}/*** 获取等待此锁的估计线程数*/public final int getQueueLength() {return sync.getQueueLength();}/*** 获取包含等待获取许可的线程集合*/protected Collection<Thread> getQueuedThreads() {return sync.getQueuedThreads();}/*** 获取标识信号量的字符串*/public String toString() {return super.toString() + "[Permits = " + sync.getPermits() + "]";}
}

semaphore示例:

package org.dsg.jdk.locks.aqs;import lombok.extern.slf4j.Slf4j;import java.util.concurrent.Semaphore;@Slf4j(topic = "c.semaphoreInstance")
public class SemaphoreInstance {public static void main(String[] args) {//1.创建信号量Semaphore semaphore = new Semaphore(3);//2.创建线程使用信号量for(int i=0;i<10;i++){new Thread(()->{try {//运行前需要获得许可semaphore.acquire();log.info("running...");Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();} finally {//使用完毕释放许可semaphore.release();}log.info("end...");}).start();}}}

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

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

相关文章

【Tomcat】Apache发布两个新版本Tomcat修复多个Bug

Apache 官网发布了两个最新的 Tomcat 版本包&#xff0c;分别是&#xff1a;8.5.94、9.0.81 这两个最新版本修复了多个漏洞&#xff0c;统计信息如下表所示。有关漏洞的详细信息&#xff0c;请查阅官方相关文档&#xff08;见&#xff1a;参考&#xff09;。 严重等级漏洞说明…

【java学习】数组中涉及的常见算法-含冒泡排序(11)

文章目录 1. 最大值、最小值、总和、平均数2. 数组的复制、反转3. 数组元素的排序3.1. 排序方式3.1.1. 冒泡排序 1. 最大值、最小值、总和、平均数 代码如下&#xff1a; public class Test1 {public static void main (String[] args) {int[] arr new int[] {4,2,7,1,3,5};/…

Element-Form表单单独取消某一项的校验

业务场景&#xff1a; **活动区域下拉框的值&#xff0c;选择为区域二时&#xff0c;显示活动形式&#xff08;必填项&#xff09;。 活动区域下拉框的值&#xff0c;选择为区域一时&#xff0c;活动形式则隐藏&#xff0c;因为活动形式为必填项&#xff0c;点保存是走ruleForm…

卷积神经网络CNN基础知识

目录 1 前言2 卷积神经网络CNN2.1 LeNet-5相关介绍2.2 CNN基本结构2.2.1 卷积层2.2.2 池化层&#xff08;下采样层&#xff09;2.2.3 全连接层2.2.3.1激励层&#xff08;非线性激活&#xff09;2.2.3.2 线性层2.2.3.3 Dropout层2.2.3.4 总结 2.3 图像的上采样和下采样2.3.1 上采…

WPF学习

1、 <ResourceDictionary.MergedDictionaries> </ResourceDictionary.MergedDictionaries> 这段XAML代码涉及到WPF中的资源字典&#xff08;Resource Dictionary&#xff09;&#xff0c;用于引入外部样式或资源以在您的WPF应用程序中重用。让我逐句解释&#xff…

联盟链学习笔记-网络的创建

联盟链学习笔记 初始网络 下图是初始网络网络N的参考图 排序服务 在定义 网络 N 的时候&#xff0c;第一件事情就是定义一个 排序服务O4。O4 最初被配置并且由组织 R4 的一个管理员来启动&#xff0c;并且由 R4 管理。配置 NC4 包含了描述网络管理能力初始集合的规则。最初…

基于vue的移动端如何监听系统返回

一、问题场景 移动端项目基于vue框架搭建的微服务子应用&#xff0c;每个小功能都是独立存在&#xff1b;当我们退出某个功能页面&#xff0c;返回主页时使用的是手机系统自带的返回功能&#xff1b;未使用程序界面左上角返回按钮&#xff0c;这时候返回逻辑中的清除缓存操作没…

微软开源 windows-drivers-rs,用 Rust 开发 Windows 驱动程序

导读Microsoft Azure 首席技术官兼著名 Windows 软件开发人员 Mark Russinovich 在社交平台上宣布&#xff0c;启动了一个名为 windows-drivers-rs 的新开源项目。 该项目可帮助开发人员使用 Rust 开发 Windows 驱动程序&#xff0c;旨在支持 WDM (Windows Driver Model) 和 W…

【C语言】程序环境和预处理

程序环境&#xff1a; 1、C语言的任何一种实现&#xff0c;存在两个不同的环境&#xff1b; 2、翻译环境&#xff1a;将源代码转换成可执行的二进制指令&#xff08;机器指令&#xff09;&#xff1b;.c文件&#xff08;源文件——文本信息的代码&#xff09;->&#xff0…

微服务12-分布式服务理论基础+Seata的认识

文章目录 分布式服务理论基础前言微服务和分布式的区别CAP定理BASE理论 Seata流程&#xff1a;seata部署微服务集成seata 分布式服务理论基础 前言 单体架构&#xff1a; 1.项目过于臃肿&#xff0c;所有服务在一起&#xff0c;一个业务挂了&#xff0c;整个项目就不能用了&…

腾讯云我的世界mc服务器多少钱一年?

腾讯云我的世界mc服务器多少钱&#xff1f;95元一年2核2G3M轻量应用服务器、2核4G5M带宽优惠价218元一年、4核8G12M带宽轻量服务器446元一年&#xff0c;云服务器CVM标准型S5实例2核2G优惠价280元一年、2核4G配置服务器748元一年&#xff0c;腾讯云百科txybk.com分享腾讯云我的…

人事管理系统springboot42

大家好✌&#xff01;我是CZ淡陌。一名专注以理论为基础实战为主的技术博主&#xff0c;将再这里为大家分享优质的实战项目&#xff0c;本人在Java毕业设计领域有多年的经验&#xff0c;陆续会更新更多优质的Java实战项目&#xff0c;希望你能有所收获&#xff0c;少走一些弯路…

Vue15 计算属性VS监视属性(侦听属性)

计算属性VS监视属性&#xff08;侦听属性&#xff09; computed和watch之间的区别&#xff1a; 1.computed能完成的功能&#xff0c;watch都可以完成。 2.watch能完成的功能&#xff0c;computed不一定能完成&#xff0c;例如&#xff1a;watch可以进行异步操作。 两个重要的小…

webpack性能优化之打包优化

1、禁止生成 sourceMap 文件 Vue打包时会生成map文件&#xff0c;map文件的作用在于&#xff1a;项目打包后&#xff0c;代码都是经过压缩加密的&#xff0c;如果运行时报错&#xff0c;输出的错误信息无法准确得知是哪里的代码报错。有了map就可以像未加密的代码一样&#xf…

HNU-计算机网络-讨论课1

题目&#xff1a; 有关网络系统结构的讨论 一、必选问题&#xff08;每组平均每人1个&#xff09; 1、分层结构其实是世界演进形成的普遍系统结构&#xff0c;不管是自然领域还是社会领域&#xff0c;请举例说说你的理解。 2、有人说通信网络的核心技术在物理层&#xff0c…

jvm打破砂锅问到底- 为什么要标记或记录跨代引用

为什么要标记或记录跨代引用. ygc时, 直接把老年代引用的新生代对象(可能是对象区域)记录下来当做根, 这其实就是依据第二假说和第三假说, 强者恒强, 跨代引用少(存在互相引用关系的两个对象&#xff0c;是应该倾 向于同时生存或者同时消亡的). 拿ygc老年代跨代引用对象当做根…

Spring Boot 中的 TransactionTemplate 是什么,如何使用

Spring Boot中的TransactionTemplate&#xff1a;简化事务管理 事务管理是任何应用程序中至关重要的部分&#xff0c;特别是在处理数据库操作时。Spring Boot提供了多种方式来管理事务&#xff0c;其中之一是使用TransactionTemplate。本文将深入探讨TransactionTemplate是什么…

设计师都应该知道的事:极简主义家具该怎么去用

这座房子有黑暗而沉重的特征&#xff0c;包括棕色和白色的马赛克浴室瓷砖&#xff0c;弯曲的锻铁壁灯和土黄色的威尼斯石膏墙。但由于房屋与他们的风格相去甚远&#xff0c;白色&#xff0c;干净和简约&#xff0c;接下来我们就着这个方向去帮助房主进行改造。 她解释说&#x…

小谈设计模式(24)—命令模式

小谈设计模式&#xff08;24&#xff09;—命令模式 专栏介绍专栏地址专栏介绍 命令模式角色分析命令&#xff08;Command&#xff09;具体命令&#xff08;ConcreteCommand&#xff09;接收者&#xff08;Receiver&#xff09;调用者&#xff08;Invoker&#xff09;客户端&am…

生物制剂\化工\化妆品等质检损耗、制造误差处理作业流程图(ODOO15/16)

生物制剂、化工、化妆品等行业&#xff0c;因为产品为液体&#xff0c;产品形态和质量容易在各个业务环节发生变化&#xff0c;常常导致实物和账面数据不一致&#xff0c;如果企业业务流程不清晰&#xff0c;会导致系统大量的库存差异&#xff0c;以及财务难以核算的问题&#…