Java信号量Semaphore的使用

时间:2022-05-03 15:10:01
Semaphore是用来保护一个或者多个共享资源的访问,Semaphore部维护了一个计数器,其值为可以访问的共享资源的个数。
一个线程要访问共享资源,先获得信号量,如果信号量的计数器值大于1,意味着有共享资源可以访问,则使其计数器值减去1,再访问共享资源。如果计数器值为0,线程进入休眠。当某个线程使用完共享资源后,释放信号量,并将信号量内部的计数器加1,之前进入休眠的线程将被唤醒并再次试图获得信号量。

一种实际的情况,大家去水房打水,只有两个水龙头,来了20个人,需要排队。

ExecutorService service = Executors.newCachedThreadPool();

Semaphore waters = new Semaphore(2); //两个水龙头
for (int i = 0; i < 20; i++) {
    service.submit(new UseWater("张三" + i, waters)); //新来的打水者
}

service.shutdown(); //所有任务执行完后关闭。

waters.acquireUninterruptibly(2);//如果水龙头前有人排队则等待线程完成。
//打水逻辑

if(number.availablePermits() > 0) {
    System.out.println(name + "正在打水");
} else {
    System.out.println(String.format("没有位置,%s-排队..", name));
}

try {
    number.acquire();//获取水龙头
    
    System.out.println(name + "开始打水..");
    Thread.sleep(((int) (100 * Math.random())));
    System.out.println(name + "打完水离开..");

    number.release();//释放水龙头

} catch (InterruptedException e) {
    e.printStackTrace();
}