一、什么是信号量

“信号量”在编程术语中使用单词semaphore,那什么是“信号量”?信号量就好比你家厨房入口架子上摆了三把锅。

  • 如果你的孩子热奶拿走一把,你的老婆热汤拿走一把,你的妈妈做菜拿走一把,你想煮面条就没有锅了。当你看到这种情况,你就不会进入厨房了,你处于等待状态。也就说厨房按照“锅的数量”作为信号量,只能容纳三个人(线程)。
  • 当你的老婆热完汤之后,把锅重新放回架子上,你就可以去获得一个锅,你就可以进入厨房了。

二、信号量类Semaphore

通过上文的介绍,我们可以总结出信号量的重要组成部分

  • 计数器:计算信号量的使用情况,锅(信号)被使用一次减1,锅(信号)被还回一次加1
  • 等待队列:当任务数量大于信号量数量上限的时候,任务进入等待队列

信号量在JDK中是由 java.util.concurrent.Semaphore 实现的,Semaphore提供了两个构造函数。permits参数代表信号量的数量(锅的数量),fair代表信号量的获取是否遵循公平原则。所谓的公平原则就是:先启动的线程先调用semaphore.acquire();方法,就先得到一个信号“锅”(permit),遵循先来后到的原则。

public Semaphore(int permits)
public Semaphore(int permits, boolean fair)

常用方法列表

方法名 作用
acquire() 获取一个permit,在获取到permit之前,线程处于阻塞状态
tryAcquire() 尝试获取一个permit,获取成功返回true,否则返回false,不阻塞线程
tryAcquire(long timeout, TimeUnit unit) 和tryAcquire()大部分实现一样,区别是提供超时设置,在超时时间范围内多次尝试,不阻塞线程。
availablePermits() 获取目前剩余的信号permit的数量
release() 释放一个permit,并唤醒一个等待信号permit的线程
hasQueuedThreads() 返回值boolean类型,判断等待队列中是否存在等待线程
getQueueLength() 获取等待队列中等待线程的数量

三、实现限流器

通过上面的介绍,我相信大家肯定可以想到Semaphore的应用场景。比如:

  • 医院门诊排号器,三个在岗医生就是3个信号permit,当超出信号量数量的时候,想就诊就只能等待
  • 停车场停车功能,n个车位就是n个信号permit,当超出信号量数量的时候,想停车也只能等待

应用场景还有很多很多,大家自己去发会创造力吧。其实无论多少种应用场景说白了:Semaphore实现的就是一个限流器。我们还是以我们家的厨房kitchen里面的三把锅wok为例,实现基于信号量的限流。

public class TestKitchenSemaphore {

  //信号量-3把锅
private static Semaphore threeWoks = new Semaphore(3); public static void main(String[] args) throws InterruptedException { //模拟5个人抢占3把锅的场景
for(int i=0;i < 5;i++){
Thread.sleep(1000); //模拟进入厨房的先后顺序,存在时间间隔 new Thread(() -> {
try {
threeWoks.acquire(); //获取一个permit,信号量计数器减1
System.out.println(Thread.currentThread().getName()
+ "拿走了一把锅,还剩" + threeWoks.availablePermits() + "把锅");
Thread.sleep(new Random().nextInt(5000)); //模拟使用锅的时长 threeWoks.release();//释放permit,信号量计数器加1
System.out.println(Thread.currentThread().getName()
+ "还回一把锅,还剩" + threeWoks.availablePermits() + "把锅"); } catch (InterruptedException e) {
e.printStackTrace();
}
}).start(); }
}
}

上文代码的输出如下,我们可以看到每acquire一次信号量减1,每release一次信号量加1。信号量的上限是3,下限是0。当达到上限的时候,只有等先占据锅permit的线程释放,其他线程才能获取到锅permit。

Thread-0拿走了一把锅,还剩2把锅
Thread-1拿走了一把锅,还剩1把锅
Thread-2拿走了一把锅,还剩0把锅 => 备注:5个线程只能获取3个锅(上限)
Thread-1还回一把锅,还剩1把锅
Thread-3拿走了一把锅,还剩0把锅 => 备注:被还回才能被再次占用,不超过3
Thread-0还回一把锅,还剩1把锅
Thread-4拿走了一把锅,还剩0把锅 => 备注:被还回才能被再次占用,不超过3
Thread-2还回一把锅,还剩1把锅
Thread-3还回一把锅,还剩2把锅
Thread-4还回一把锅,还剩3把锅 => 备注:用完依次释放

欢迎关注我的博客,更多精品知识合集

本文转载注明出处(必须带连接,不能只转文字):字母哥博客 - zimug.com

觉得对您有帮助的话,帮我点赞、分享!您的支持是我不竭的创作动力!。另外,笔者最近一段时间输出了如下的精品内容,期待您的关注。

[java并发编程]基于信号量semaphore实现限流器的更多相关文章

  1. Java并发编程笔记之Semaphore信号量源码分析

    JUC 中 Semaphore 的使用与原理分析,Semaphore 也是 Java 中的一个同步器,与 CountDownLatch 和 CycleBarrier 不同在于它内部的计数器是递增的,那 ...

  2. Java并发编程--5.信号量和障碍器

    Semaphore信号量 简介 它本质上是一个共享锁,限制访问公共资源的线程数目,它也被称为计数信号量acquire()许可一个线程, Semaphore – 1; 没有可用的许可时,Semaphor ...

  3. Java并发编程系列之Semaphore详解

    简单介绍 我们以饭店为例,假设饭店只有三个座位,一开始三个座位都是空的.这时如果同时来了三个客人,服务员人允许他们进去用餐,然后对外说暂无座位.后来的客人必须在门口等待,直到有客人离开.这时,如果有一 ...

  4. 【Java并发编程实战】-----“J.U.C”:Semaphore

    信号量Semaphore是一个控制访问多个共享资源的计数器,它本质上是一个"共享锁". Java并发提供了两种加锁模式:共享锁和独占锁.前面LZ介绍的ReentrantLock就是 ...

  5. Java并发编程的4个同步辅助类(CountDownLatch、CyclicBarrier、Semaphore、Phaser)

    我在<JDK1.5引入的concurrent包>中,曾经介绍过CountDownLatch.CyclicBarrier两个类,还给出了CountDownLatch的演示案例.这里再系统总结 ...

  6. Java并发编程:CountDownLatch、CyclicBarrier和Semaphore

    Java并发编程:CountDownLatch.CyclicBarrier和Semaphore 在java 1.5中,提供了一些非常有用的辅助类来帮助我们进行并发编程,比如CountDownLatch ...

  7. Java并发编程Semaphore

    信号量 信号量类Semaphore,用来保护对唯一共享资源的访问.一个简单的打印队列,并发任务进行打印,加入信号量同时之能有一个线程进行打印任务 . import java.util.concurre ...

  8. 【Java并发编程五】信号量

    一.概述 技术信号量用来控制能够同时访问某特定资源的活动的数量,或者同时执行某一给定操作的数据.计数信号量可以用来实现资源池或者给一个容器限定边界. 信号量维护了一个许可集,许可的初始量通过构造函数传 ...

  9. 14、Java并发编程:CountDownLatch、CyclicBarrier和Semaphore

    Java并发编程:CountDownLatch.CyclicBarrier和Semaphore 在java 1.5中,提供了一些非常有用的辅助类来帮助我们进行并发编程,比如CountDownLatch ...

随机推荐

  1. 插值方法 - Lagrange插值多项式

    Lagrange插值多项式代码: 1 # -*- coding: utf-8 -*- 2 """ 3 Created on Wed Mar 25 15:43:42 202 ...

  2. React+Webpack+ES6 兼容低版本浏览器(IE9)解决方案

    虽然过了兼容IE6的噩梦时代,IE依旧阴魂不散,因为你可能还要兼容IE9.在ES6已经普及的今天,用ES6写react已经成了标配.但是babel编译的js语法,由于某些不规范的写法,可能在IE9下不 ...

  3. canvas菜鸟基于小程序实现图案在线定制功能

    前言 最近收到一个这样的需求,要求做一个基于 vue 和 element-ui 的通用后台框架页,具体要求如下: 要求通用性高,需要在后期四十多个子项目中使用,所以大部分地方都做成可配置的. 要求做成 ...

  4. 【Android开发】【数据库】LitePal 数据库的使用

    一,导包 dependencies { ...... // LitePal的包 compile 'org.litepal.android:core:1.3.1' ...... } 二,创建bean类 ...

  5. Mxnet速查_CPU和GPU的mnist预测训练_模型导出_模型导入再预测_导出onnx并预测

    需要做点什么 方便广大烟酒生研究生.人工智障炼丹师算法工程师快速使用mxnet,所以特写此文章,默认使用者已有基本的深度学习概念.数据集概念. 系统环境 python 3.7.4 mxnet 1.9. ...

  6. 免费的天气API

    高德地图天气 天气查询-API文档 请求示例: { "status": "1", "count": "1", " ...

  7. Java学习day39

    类加载的作用:将class文件字节码内容加载到内存中,并将这些静态数据转换成方法区的运行时数据结构,然后在堆中生成一个代表这个类的java.lang.Class对象,作为方法区中类数据的访问入口. 类 ...

  8. Spring Authorization Server授权服务器入门

    11月8日Spring官方已经强烈建议使用Spring Authorization Server替换已经过时的Spring Security OAuth2.0,距离Spring Security OA ...

  9. Git删除已提交的文件

    Git删除已提交的文件 Git删除已提交的文件 定位文件 删除文件 参考链接 昨天通过Git Bash提交代码的时候遇到了由于单个文件大小超过100M,导致代码上传失败的问题.考虑到那个大文件是用于训 ...

  10. 【ASP.NET Core】自己编程来生成自签名的服务器证书

    如果项目不大,或者是客户公司内部使用,或者不想花钱购买证书,又或者用于开发阶段测试--完全可以使用自签名证书. 所谓自签,就是自己给自己签名颁发的证书,自给自足,丰衣足食. 生成证书的方法和工具很多, ...