信号量 Semaphore

西魏陶渊明 ... 2021-12-25 Java进阶 大约 2 分钟

作者: 西魏陶渊明 博客: https://blog.springlearn.cn/ (opens new window)

西魏陶渊明

莫笑少年江湖梦,谁不少年梦江湖

# 🚀 知识快读

Semaphore 翻译过来就是信号量, 其根本原理就是基于 CAS 共享锁的一种实现。举一个例子。 假设停车场只有三个车位,一开始三个车位都是空的。这时如果同时来了五辆车,看门人允许其中三辆不受阻碍的进入,然后放下车拦,剩下的车则必须在入口等待,此后来的车也都不得不在入口处等待。这时,有一辆车离开停车场,看门人得知后,打开车拦,放入一辆,如果又离开两辆,则又可以放入两辆,如此往复。

那么上面的这个例子可以这样理解,资源一共有3个, 即三个车位。如何来控制这5辆汽车,来合理的使用这3个资源呢? Semaphore 可以这样来定义。

// 1. 定一个信号量,声明有3个资源。使用公平模式线程将会按到达的顺序(FIFO)执行(也就是等待时间最长的先执行),如果是非公平,则可以后请求的有可能排在队列的头部。
Semaphore semp = new Semaphore(3);
// 2. 获取1个许可 - 最大允许3个进入,一但超过就让其等待,除非已经释放
semp.acquire();  
// 3. 释放1个许可 
semp.release(); 
// 4. 获取1许可,失败就返回,不等待
semp.tryAcquire();  
// 5. 获取2许可,失败就返回,不等待
semp.tryAcquire(2);  
// 6. 不允许被中断
semp.acquireUninterruptibly();
1
2
3
4
5
6
7
8
9
10
11
12

# 知识点1: Fair & NoFair

Semaphore 的模式配置,只是构造来定义。

  • 默认构造不公平模式, 谁来申请资源,就先尝试获取资源。排队的要等到没有资源进来申请才能继续申请
    public Semaphore(int permits) {
        sync = new NonfairSync(permits);
    }

    public Semaphore(int permits, boolean fair) {
        sync = fair ? new FairSync(permits) : new NonfairSync(permits);
    }
1
2
3
4
5
6
7

# 知识点2: 申请资源

  • acquire() 获取1个资源,获取不到就等待,如果线程中断,会直接中断。
  • acquire(2) 获取2个资源,获取不到就等待,如果线程中断,会直接中断。
  • tryAcquire() 获取1个资源,获取不到就返回 false,如果线程中断,会直接中断。
  • acquireUninterruptibly() 获取1个资源,获取不到就等待,不会关心线程中断。

# 知识点3: 释放资源

  • release() 释放一个资源
  • release(2) 释放两个资源

# 知识点4: 其他API

  • availablePermits() 当前资源数量
  • drainPermits() 获取当前资源数量,并将剩余资源清零,直接赋值0
  • reducePermits(2) 将资源数量,扣减2个
  • isFair() 是否公平
  • hasQueuedThreads() 是否还有线程等待
  • getQueueLength() 还有多少线程等待
  • getQueuedThreads() 获取所有的线程集合

本文由西魏陶渊明版权所有。如若转载,请注明出处:西魏陶渊明
上次编辑于: 2022年6月16日 21:10
贡献者: lxchinesszz