面试官:说说Java中的信号量?Semaphore

Semaphore (信号量)是由计算机科学家Dijkstra在1965年提出的,普遍应用不一样的操做系统,在管程提出以前信号量就是并发编程领域的霸主!几乎全部并发的语言都支持信号量机制。java

Semaphore 也有被翻译成信号灯,由于其机制就像咱们平常生活中的红绿灯,车辆的通行看红绿灯,对应编程世界的线程能不能执行得看信号灯!面试

Semaphore 用来多线程互斥问题,相对于synchronized和Lock来讲它容许多个线程访问一个临界区!例如各类池:数据库链接池、对象池等,这些池的需求就是同一时刻容许多个线程同时使用链接池。数据库

Semaphore的模型能够归纳为一个计数器、一个等待队列、三个方法。 三个方法原子性分别是init()、down()、up()编程

init():设置计数器的初始值。bash

down():将计数器的值减一,若是减了一以后,计数器的值小于0,则当前的线程被阻塞,不然继续执行。多线程

up():将计数器的值加一,若是加了一以后,计数器的值小于等于0,则唤醒等待队列中的一个线程,而且将它移除出等待队列。(注意是小于等于0,不该该理解为大于等于0,由于大于等于0代表此时没有等待的线程,因此不会有唤醒这个操做。) 并发

简单的理解就是 Semaphore 就是经过这三个方法来改变计数器,经过计数器的值来判断此时的线程是应该加入到等待队列中等待仍是成功执行。分布式

信号量模型也被称为PV原语,也就是downup操做最先称为P操做和V操做,有些人还称为semWaitsemSignal在JAVA中信号量模型是由 java.util.concurrent.Semaphore 的实现,而且downup对应的实现方法是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三个类,NonfairSyncFairSync类继承自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)资料等待领取

相关文章
相关标签/搜索