AQS系列(六)- Semaphore的使用及原理
前言
Semaphore也是JUC包中一个用于并发控制的工具类,举个常用场景的例子:有三台电脑五个人,每个人都要用电脑注册一个自己的账户,这时最开始只能同时有三个人操作电脑注册账户,这三个人中有人操作完了剩下的两个人才能占用电脑注册自己的账户。这就是Semaphore的经典使用场景,跟并发加锁有点像,只是我们的并发加锁同一时间只让有一个线程执行,而Semaphore的加锁控制是允许同一时间有指定数量的线程同时执行,超过这个数量就加锁控制。
一、使用样例
1 public static void main(String[] args) { 2 Semaphore semaphore = new Semaphore(3); // 对比上面例子中的3台电脑 3 for (int i = 0; i < 5; i++) { // 对比上面例子中的5个人 4 new Thread(() -> { 5 try { 6 semaphore.acquire(1); // 注意acquire中的值可以传任意值>=0的整数 7 } catch (InterruptedException e) { 8 e.printStackTrace(); 9 } 10 System.out.println(Thread.currentThread().getName() + " acquire 1"); 11 try { 12 Thread.sleep(2000); 13 } catch (InterruptedException e) { 14 e.printStackTrace(); 15 } 16 System.out.println(Thread.currentThread().getName() + "release 1"); 17 semaphore.release(1); 18 }).start(); 19 } 20 }
执行结果为:
1 Thread-0 acquire 1 2 Thread-2 acquire 1 3 Thread-1 acquire 1 4 Thread-1release 1 5 Thread-2release 1 6 Thread-0release 1 7 Thread-4 acquire 1 8 Thread-3 acquire 1 9 Thread-4release 1 10 Thread-3release 1
可以看到同一时间只有三个线程获取到了锁,这三个执行完释放了之后,剩下两个菜获取锁执行。下面看看源码是如何实现的。
二、源码实现
1、Semaphore构造器
1 public Semaphore(int permits) { 2 sync = new NonfairSync(permits); 3 } 4 5 public Semaphore(int permits, boolean fair) { 6 sync = fair ? new FairSync(permits) : new NonfairSync(permits); 7 }
可以看到,Semaphore有两个构造器,一个是只传数值默认非公平锁,另一个可指定用公平锁还是非公平锁。permits最终还是赋值给了AQS中的state变量。
2、acquire(1)方法
1 public void acquire(int permits) throws InterruptedException { 2 if (permits < 0) throw new IllegalArgumentException(); 3 sync.acquireSharedInterruptibly(permits); 4 }
此方法同样调用了AQS中的模板方法:
1 public final void acquireSharedInterruptibly(int arg) 2 throws InterruptedException { 3 if (Thread.interrupted()) 4 throw new InterruptedException(); 5 if (tryAcquireShared(arg) < 0) 6 doAcquireSharedInterruptibly(arg); 7 }
1)、查看tryAcquireShared的实现方法
先看非公平锁的获取:
1 final int nonfairTryAcquireShared(int acquires) { 2 for (;;) { 3 int available = getState(); 4 int remaining = available - acquires; // 如果remaining是负的,说明当前剩余的信号量不够了,需要阻塞 5 if (remaining < 0 || 6 compareAndSetState(available, remaining)) // 如果remaining<0则直接return,不会走CAS;如果大于0,说明信号量还够,可走CAS将信号量减掉,成功则返回大于0的remaining 7 return remaining; 8 } 9 }
再看公平锁的获取:
1 protected int tryAcquireShared(int acquires) { 2 for (;;) { 3 if (hasQueuedPredecessors()) // 判断是不是在队首,不是的话直接返回-1 4 return -1; 5 int available = getState(); // 后面逻辑通非公平锁的获取逻辑 6 int remaining = available - acquires; 7 if (remaining < 0 || 8 compareAndSetState(available, remaining)) 9 return remaining; 10 } 11 }
可以看到,不管非公平锁和公平锁,加锁时都是先判断当前state够不够减的,如果减出负数返回获取锁失败,是正数才走CAS将原信号量扣掉,返回获取锁成功。加锁时一个减state的过程。
2)、doAcquireSharedInterruptibly
此方法还是AQS中的实现,逻辑重复,就不再说明了。
3、release(1)方法
1 public void release(int permits) { 2 if (permits < 0) throw new IllegalArgumentException(); 3 sync.releaseShared(permits); 4 }
同样调用了AQS中的模板方法releaseShared:
1 public final boolean releaseShared(int arg) { 2 if (tryReleaseShared(arg)) { 3 doReleaseShared(); 4 return true; 5 } 6 return false; 7 }
其中tryReleaseShared的实现在Semaphore类的Sync中,如下所示:
1 protected final boolean tryReleaseShared(int releases) { 2 for (;;) { 3 int current = getState(); 4 int next = current + releases; // 用当前state加上要释放的releases 5 if (next < current) // overflow 6 throw new Error("Maximum permit count exceeded"); 7 if (compareAndSetState(current, next)) // 用CAS将state加上 8 return true; 9 } 10 }
另一个方法doReleaseShared之前看过,此处就不赘述了。
三、小结
Semaphore信号量类基于AQS的共享锁实现,有公平锁和非公平锁两个版本。它的加锁与释放锁的不同之处在于和普通的加锁释放锁反着,ReentrantLock和ReentrantReadWriteLock中都是加锁时state+1,释放锁时state-1,而Semaphore中是加锁时state减,释放锁时state加。
另外,如果它还可以acquire(2) 、release(1),即获取的和释放的信号量可以不一致,只是需要注意别释放的信号量太少导致后续任务获取不到足够的量而永久阻塞。