Semaphore
(信号量)是由计算机科学家Dijkstra在1965年提出的,普遍应用不一样的操做系统,在管程提出以前信号量就是并发编程领域的霸主!几乎全部并发的语言都支持信号量机制。java
Semaphore
也有被翻译成信号灯,由于其机制就像咱们平常生活中的红绿灯,车辆的通行看红绿灯,对应编程世界的线程能不能执行得看信号灯!面试
Semaphore
用来多线程互斥问题,相对于synchronized和Lock来讲它容许多个线程访问一个临界区!例如各类池:数据库链接池、对象池等,这些池的需求就是同一时刻容许多个线程同时使用链接池。数据库
Semaphore的模型能够归纳为一个计数器、一个等待队列、三个方法。 三个方法原子性分别是init()、down()、up()
;编程
init()
:设置计数器的初始值。bash
down()
:将计数器的值减一,若是减了一以后,计数器的值小于0,则当前的线程被阻塞,不然继续执行。多线程
up()
:将计数器的值加一,若是加了一以后,计数器的值小于等于0,则唤醒等待队列中的一个线程,而且将它移除出等待队列。(注意是小于等于0,不该该理解为大于等于0,由于大于等于0代表此时没有等待的线程,因此不会有唤醒这个操做。) 并发
简单的理解就是 Semaphore
就是经过这三个方法来改变计数器,经过计数器的值来判断此时的线程是应该加入到等待队列中等待仍是成功执行。分布式
信号量模型也被称为PV原语,也就是down
和up
操做最先称为P操做和V操做,有些人还称为semWait
和semSignal
。在JAVA中信号量模型是由 java.util.concurrent.Semaphore
的实现,而且down
和up
对应的实现方法是acquire和release
,咱们来看下简单的使用例子函数
int count;
final Semaphore semaphore = new Semaphore(1); // 初始化信号量
// 用信号量保证互斥
void addOne() {
try {
semaphore.acquire(); //对应down
count+=1;
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
semaphore.release(); //对应up
}
}
复制代码
若是你想多让几个线程进去临界区,那么就把Semaphore
构造器中的1改成你想要的线程数。性能
能够理解为颁发许可证,好比想同时容许3个线程进入临界区,构造器中的数就填3,理解为搞了3张许可证,而后颁发出去,谁拿到了许可证谁就能进临界区,进入临界区后的线程搞完事了,就归还许可证,而后出去。
Semaphore
的内部共存在Sync、NonfairSync、FairSync
三个类,NonfairSync
与FairSync
类继承自Sync
类,Sync
类继承自AbstractQueuedSynchronizer
抽象类,也就Semaphore
是依托于NonfairSync、FairSync
来实现的。
public Semaphore(int permits) {
sync = new NonfairSync(permits);
}
public Semaphore(int permits, boolean fair) {
sync = fair ? new FairSync(permits) : new NonfairSync(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;
}
}
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;
}
}
复制代码
差异就在于有没hasQueuedPredecessors()
,这个方法就是判断当前线程是不是等待队列中的头结点,若是不是,则不给于分配。 大体Semaphore
的模型和模型实现思路就是这样,建议多看看源码,不难的能够加深理解,而且懂得具体实现以后能掌握把控更多细节,还不怕面试官问。
若有错误欢迎指正!
我的公众号:yes的练级攻略
有相关面试进阶(分布式、性能调优、经典书籍pdf)资料等待领取