java 信号量
我们知道可以通过信号量控制共享资源的访问,底层还是AQS这一套,这没什么难的。但是有一点可能被大家忽略:声明信号量的时候,比如只有3个许可证,但是运行过程中,某个时刻的许可证数量是没有限制的。
比较变态的一点
Semaphore semaphore = new Semaphore(3);
Runnable runnable = () -> {
try {
System.out.println(Thread.currentThread().getName() + "try acquire");
semaphore.acquire();
System.out.println(Thread.currentThread().getName() + "acquired semaphore");
} catch (InterruptedException e) {
e.printStackTrace();
}
};
semaphore.release();
// 4个线程都能够获取到信号量
for (int i = 0; i < 4; i++) {
new Thread(runnable).start();
}
即使创建信号量的时候,指定了信号量的大小。但是在通过 release()操作释放信号量任然能超过配置的大小。也就有可能同时执行的线程数量比最开始设置的要大。
没有任何线程获取信号量的时候,依然能够释放并且释放的有效。
推荐的做法是一个线程先 acquire 然后 release。如果释放线程和获取线程不是同一个,那么最好保证这种对应关系。不要释放过多的许可证。