semaphore [ˈseməfɔː(r)] 发信号;信号标;旗语
Semaphore 信号量
Semaphore是一种基于计数的信号量。
它可以设定一个阈值,基于此,多个线程竞争获取许可信号acquire(),完成操作后归还信号release()。超过阈值后,线程申请许可信号将会被阻塞。
Semaphore可以用来构建一些对象池,资源池之类的,比如数据库连接池,我们也可以创建计数为1的Semaphore,将其作为一种类似互斥锁的机制,这也叫二元信号量,表示两种互斥状态。
下面的Demo中申明了一个只有5个许可的Semaphore,而有20个线程要访问这个资源,通过acquire()和release()获取和释放访问许可。
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Semaphore;
public class TestSemaphore {
public static void main(String[] args) {
// 线程池
ExecutorService exec = Executors.newCachedThreadPool();
// 只能5个线程同时访问
final Semaphore semp = new Semaphore(5);
// 模拟20个客户端访问
for (int index = 1; index <= 20; index++) {
final int NO = index;
Runnable run = new Runnable() {
public void run() {
try {
// 获取许可
semp.acquire();
System.out.println("Accessing: " + NO + " 进入");
Thread.sleep((long) (Math.random() * 10000));
// 访问完后,释放
semp.release();
System.out.println("Accessing: " + NO + " 离开!==>");
System.out.println("-----------------" + semp.availablePermits());
} catch (InterruptedException e) {
e.printStackTrace();
}
}
};
exec.execute(run);
}
// 退出线程池
exec.shutdown();
}
运行结果:(多线程下,每次运行结果都不一样,仅供参考)
Accessing: 2 进入
Accessing: 1 进入
Accessing: 4 进入
Accessing: 5 进入
Accessing: 3 进入
Accessing: 3 离开!==>
Accessing: 6 进入
-----------------0
Accessing: 4 离开!==>
Accessing: 7 进入
-----------------0
Accessing: 1 离开!==>
-----------------0
Accessing: 8 进入
Accessing: 9 进入
Accessing: 2 离开!==>
-----------------0
Accessing: 5 离开!==>
-----------------0
Accessing: 10 进入
Accessing: 6 离开!==>
Accessing: 11 进入
-----------------0
Accessing: 9 离开!==>
Accessing: 12 进入
-----------------0
Accessing: 10 离开!==>
-----------------0
Accessing: 18 进入
Accessing: 7 离开!==>
Accessing: 16 进入
-----------------0
Accessing: 18 离开!==>
-----------------0
Accessing: 17 进入
Accessing: 8 离开!==>
-----------------0
Accessing: 13 进入
Accessing: 12 离开!==>
Accessing: 20 进入
-----------------0
Accessing: 16 离开!==>
-----------------0
Accessing: 19 进入
Accessing: 11 离开!==>
Accessing: 15 进入
-----------------0
Accessing: 17 离开!==>
-----------------0
Accessing: 14 进入
Accessing: 20 离开!==>
-----------------1
Accessing: 13 离开!==>
-----------------2
Accessing: 15 离开!==>
-----------------3
Accessing: 19 离开!==>
-----------------4
Accessing: 14 离开!==>
-----------------5