作者:chenssy

来源:Java技术栈公众号

信号量Semaphore是一个控制访问多个共享资源的计数器,和CountDownLatch一样,其本质上是一个“共享锁”。

Semaphore,在API是这么介绍的:一个计数信号量。

从概念上讲,信号量维护了一个许可集。如有必要,在许可可用前会阻塞每一个 acquire(),然后再获取该许可。每个 release() 添加一个许可,从而可能释放一个正在阻塞的获取者。但是,不使用实际的许可对象,Semaphore 只对可用许可的号码进行计数,并采取相应的行动。

Semaphore 通常用于限制可以访问某些资源(物理或逻辑的)的线程数目。

下面我们就一个停车场的简单例子来阐述Semaphore:

为了简单起见我们假设停车场仅有5个停车位,一开始停车场没有车辆所有车位全部空着,然后先后到来三辆车,停车场车位够,安排进去停车,然后又来三辆,这个时候由于只有两个停车位,所有只能停两辆,其余一辆必须在外面候着,直到停车场有空车位,当然以后每来一辆都需要在外面候着。当停车场有车开出去,里面有空位了,则安排一辆车进去(至于是哪辆 要看选择的机制是公平还是非公平)。

从程序角度看,停车场就相当于信号量Semaphore,其中许可数为5,车辆就相对线程。当来一辆车时,许可数就会减 1 ,当停车场没有车位了(许可书 == 0 ),其他来的车辆需要在外面等候着。如果有一辆车开出停车场,许可数 + 1,然后放进来一辆车。

号量Semaphore是一个非负整数(>=1)。当一个线程想要访问某个共享资源时,它必须要先获取Semaphore,当Semaphore >0时,获取该资源并使Semaphore – 1。如果Semaphore值 = 0,则表示全部的共享资源已经被其他线程全部占用,线程必须要等待其他线程释放资源。当线程释放资源时,Semaphore则+1

1、实现分析

Semaphore结构如下:

从上图可以看出Semaphore内部包含公平锁(FairSync)和非公平锁(NonfairSync),继承内部类Sync,其中Sync继承AQS(再一次阐述AQS的重要性)。

Semaphore提供了两个构造函数:

  1. Semaphore(int permits) :创建具有给定的许可数和非公平的公平设置的 Semaphore。

  2. Semaphore(int permits, boolean fair) :创建具有给定的许可数和给定的公平设置的 Semaphore。

实现如下:

    public Semaphore(int permits) {sync = new NonfairSync(permits);}public Semaphore(int permits, boolean fair) {sync = fair ? new FairSync(permits) : new NonfairSync(permits);}

Semaphore默认选择非公平锁。

当信号量Semaphore = 1 时,它可以当作互斥锁使用。其中0、1就相当于它的状态,当=1时表示其他线程可以获取,当=0时,排他,即其他线程必须要等待。

2、信号量获取

Semaphore提供了acquire()方法来获取一个许可。

    public void acquire() throws InterruptedException {sync.acquireSharedInterruptibly(1);}

内部调用AQS的acquireSharedInterruptibly(int arg),该方法以共享模式获取同步状态:

    public final void acquireSharedInterruptibly(int arg)throws InterruptedException {if (Thread.interrupted())throw new InterruptedException();if (tryAcquireShared(arg) < 0)doAcquireSharedInterruptibly(arg);}

在acquireSharedInterruptibly(int arg)中,tryAcquireShared(int arg)由子类来实现,对于Semaphore而言,如果我们选择非公平模式,则调用NonfairSync的tryAcquireShared(int arg)方法,否则调用FairSync的tryAcquireShared(int arg)方法。

3、公平

    protected int tryAcquireShared(int acquires) {for (;;) {//判断该线程是否位于CLH队列的列头if (hasQueuedPredecessors())return -1;//获取当前的信号量许可int available = getState();//设置“获得acquires个信号量许可之后,剩余的信号量许可数”int remaining = available - acquires;//CAS设置信号量if (remaining < 0 ||compareAndSetState(available, remaining))return remaining;}}

4、非公平

对于非公平而言,因为它不需要判断当前线程是否位于CLH同步队列列头,所以相对而言会简单些。

        protected int tryAcquireShared(int acquires) {return nonfairTryAcquireShared(acquires);}final int nonfairTryAcquireShared(int acquires) {for (;;) {int available = getState();int remaining = available - acquires;if (remaining < 0 ||compareAndSetState(available, remaining))return remaining;}}
5、信号量释放

获取了许可,当用完之后就需要释放,Semaphore提供release()来释放许可。

    public void release() {sync.releaseShared(1);}

内部调用AQS的releaseShared(int arg):

    public final boolean releaseShared(int arg) {if (tryReleaseShared(arg)) {doReleaseShared();return true;}return false;}

releaseShared(int arg)调用Semaphore内部类Sync的tryReleaseShared(int arg):

    protected final boolean tryReleaseShared(int releases) {for (;;) {int current = getState();//信号量的许可数 = 当前信号许可数 + 待释放的信号许可数int next = current + releases;if (next < current) // overflowthrow new Error("Maximum permit count exceeded");//设置可获取的信号许可数为nextif (compareAndSetState(current, next))return true;}}

6、应用示例

我们已停车为示例:

  1. public class SemaphoreTest{

  2. static class Parking{

  3.    //信号量

  4.    private Semaphore semaphore;

  5.    Parking(int count){

  6.        semaphore = new Semaphore(count);

  7.    }

  8.    public void park(){

  9.        try {

  10.            //获取信号量

  11.            semaphore.acquire();

  12.            long time = (long) (Math.random() * 10);

  13.            System.out.println(Thread.currentThread().getName() + "进入停车场,停车" + time + "秒..." );

  14.            Thread.sleep(time);

  15.            System.out.println(Thread.currentThread().getName() + "开出停车场...");

  16.        } catch (InterruptedException e) {

  17.            e.printStackTrace();

  18.        } finally {

  19.            semaphore.release();

  20.        }

  21.    }

  22. }

  23. static class Car extends Thread {

  24.    Parking parking ;

  25.    Car(Parking parking){

  26.        this.parking = parking;

  27.    }

  28.    @Override

  29.    public void run() {

  30.        parking.park();     //进入停车场

  31.    }

  32. }

  33. public static void main(String[] args){

  34.    Parking parking = new Parking(3);

  35.    for(int i = 0 ; i < 5 ; i++){

  36.        new Car(parking).start();

  37.    }

  38. }


运行结果如下:


- END -

 往期推荐:

  • 死磕Java系列:

  1. 深入分析ThreadLocal

  2. 深入分析synchronized的实现原理

  3. 深入分析volatile的实现原理

  4. Java内存模型之happens-before

  5. Java内存模型之重排序

  6. Java内存模型之分析volatile

  7. Java内存模型之总结

  8. J.U.C之AQS简介

  9. J.U.C之AQS:CLH同步队列

  10. J.U.C之AQS同步状态的获取与释放

  11. J.U.C之AQS阻塞和唤醒线程

  12. J.U.C之重入锁:ReentrantLock

……

  • Spring系列:

  1. Spring Cloud Zuul中使用Swagger汇总API接口文档

  2. Spring Cloud Config Server迁移节点或容器化带来的问题

  3. Spring Cloud Config对特殊字符加密的处理

  4. Spring Boot使用@Async实现异步调用:使用Future以及定义超时

  5. Spring Cloud构建微服务架构:分布式配置中心(加密解密)

  6. Spring Boot快速开发利器:Spring Boot CLI

……

可关注我的公众号

深入交流、更多福利

扫码加入我的知识星球

点击“阅读原文”,看本号其他精彩内容

死磕Java并发:J.U.C之并发工具类:Semaphore相关推荐

  1. 死磕Java并发:J.U.C之阻塞队列:ArrayBlockingQueue

    作者:chenssy 来源:Java技术驿站 ArrayBlockingQueue,一个由数组实现的有界阻塞队列.该队列采用FIFO的原则对元素进行排序添加的. ArrayBlockingQueue为 ...

  2. 死磕Java并发:J.U.C之Java并发容器:ConcurrentLinkedQueue

    作者:chenssy 来源:Java技术驿站 要实现一个线程安全的队列有两种方式:阻塞和非阻塞.阻塞队列无非就是锁的应用,而非阻塞则是CAS算法的应用.下面我们就开始一个非阻塞算法的研究:Coucur ...

  3. 死磕Java并发:J.U.C之并发工具类:Exchanger

    作者:chenssy 来源:Java技术驿站 前面三篇博客分别介绍了CyclicBarrier.CountDownLatch.Semaphore,现在介绍并发工具类中的最后一个Exchange.Exc ...

  4. 死磕Java并发:J.U.C之并发工具类:CountDownLatch

    作者:chenssy 来源:Java技术驿站 在上篇博客中介绍了Java四大并发工具一直的CyclicBarrier,今天要介绍的CountDownLatch与CyclicBarrier有点儿相似. ...

  5. 死磕Java并发:J.U.C之Condition

    作者:chenssy 来源:http://cmsblogs.com/?p=2222 在没有Lock之前,我们使用synchronized来控制同步,配合Object的wait().notify()系列 ...

  6. 死磕Java并发:J.U.C之读写锁:ReentrantReadWriteLock

    作者:chenssy 来源:http://cmsblogs.com/?p=2213 重入锁ReentrantLock是排他锁,排他锁在同一时刻仅有一个线程可以进行访问,但是在大多数场景下,大部分时间都 ...

  7. 死磕Java并发:J.U.C之AQS:CLH同步队列

    本文转载自公号:Java技术驿站 在上篇文章"死磕Java并发:J.U.C之AQS简介"中提到了AQS内部维护着一个FIFO队列,该队列就是CLH同步队列. CLH同步队列是一个F ...

  8. 死磕Java并发:J.U.C之AQS简介

    本文转载自公众号: Java技术驿站 Java的内置锁一直都是备受争议的,在JDK 1.6之前,synchronized这个重量级锁其性能一直都是较为低下,虽然在1.6后,进行大量的锁优化策略(死磕J ...

  9. 【死磕Java并发】-----J.U.C之AQS:CLH同步队列

    原文出处:https://www.cmsblogs.com/category/1391296887813967872 『chenssy』 在上篇博客[死磕Java并发]-----J.U.C之AQS:A ...

最新文章

  1. 卫星图像中的车辆分析--A Large Contextual Dataset for Classification, Detection and Counting of Cars
  2. [杭电ACM]3336Count the string
  3. php curl的数据后台如何接收,PHP curl以模拟put请求,后台无法接受到数据是怎么回事?...
  4. 车牌识别中的可识性、放弃率、精度与识别率辨析
  5. 二叉树C++ | 链表递归实现二叉树(插入、搜索)_1
  6. 【转】sqlserver使用sql导出索引
  7. HDU2566 统计硬币【暴力】
  8. c语言函数返回数组_C语言如何用一维数组拷贝函数,拷贝二位数组(C Primer Plus 10-7)...
  9. CC2530串口中断接收字符并且判断
  10. 和威金作者的聊天记录 (附:VIKING专杀及杀毒思路)
  11. LTspice绘制方波
  12. Python给Word加水印
  13. mac重启php-fpm
  14. anaconda linux环境变量,配置anaconda环境(linux)
  15. 计算机类图书按中图法类号,中图分类号 中国图书馆分类法(O类 数理科学和化学)...
  16. k、K、kb、kB、Kb、KB、GB、MB之间的关系
  17. dameware远程服务器进不去,关于DameWare无法进行远程连接的处理方法
  18. 【云原生】2.1 Kubernetes基础概念
  19. 如何将MP4转换为MP3?四种简单易行的方法!
  20. 打怪升级之小白的大数据之旅(四十六)<HDFS各模块的原理>

热门文章

  1. golang range 遍历 索引和值
  2. windows测试模式打开关闭
  3. php 数组 添加元素、删除元素
  4. vs内存泄露检测方法
  5. C将十六进制数字字符串转成数字
  6. Visual Studio调试之符号文件
  7. eclipse运行android项目出现The connection to adb is down, and a severe error has occured.的问题
  8. 常见算法在实际项目中的应用
  9. thinkphp读写mysql的枷锁_thinkphp5 数据库配置读写分离
  10. fileZilla连接oracle服务器,传DMP文件