Semaphore(信号量)从概念上讲,信号量维护了一个许可集。如有必要,在许可可用前会阻塞每一个 acquire(),然后再获取该许可。每个 release() 添加一个许可,从而可能释放一个正在阻塞的获取者。但是,不使用实际的许可对象,Semaphore 只对可用许可的号码进行计数,并采取相应的行动。拿到信号量的线程可以进入代码,否则就等待。通过acquire()和release()获取和释放访问许可。
Semaphore semaphore = new Semaphore(5);
for (int i = 0; i < 9; i++) {
new Thread(() -> {
try {
semaphore.acquire();
System.out.println(Thread.currentThread().getName()+"\t"+"获取到了"+System.currentTimeMillis());
TimeUnit.SECONDS.sleep(1);
System.out.println(Thread.currentThread().getName()+"\t"+"要释放了"+System.currentTimeMillis());
semaphore.release();
System.out.println(Thread.currentThread().getName()+"\t"+"释放完了"+System.currentTimeMillis());
} catch (InterruptedException e) {
e.printStackTrace();
}
}, String.valueOf(i)).start();
}
CountDownLatch,CyclicBarrier,Semaphore的区别
CountDownLatch和CyclicBarrier都能够实现线程之间的等待,只不过它们侧重点不同。
CountDownLatch一般用于某个线程等待若干个其他线程执行完任务之后,它才执行。
CyclicBarrier一般用于一组线程互相等待至某个状态,然后这一组线程再同时执行;
CountDownLatch是不能够重用的,而CyclicBarrier是可以重用的。
Semaphore其实和锁有点类似,它一般用于控制对某组资源的访问权限。