1、概述
Semaphore的形参内设置的变量值代表可以处理任务的数量,和线程池有点类似。
2、使用场景
好处:好处就是可以限制并发的数量。这个线程池也可以做到。
3、Semaphore的不足之处
- 可能导致死锁:如果一个线程持有多个Semaphore许可并在获取其他许可时阻塞,同时其他线程也在尝试获取这些许可,这就可能导致死锁。在使用Semaphore时,需要注意避免死锁问题。
- 无法控制锁的顺序:Semaphore不能控制获取许可的线程顺序,可能导致一些线程被长时间阻塞,而其他线程持续获取许可。这种情况下,可以考虑使用其他同步工具类,如ReentrantLock和Condition。
- 不支持读写锁:Semaphore不能区分读写操作,如果需要实现读写锁功能,可以考虑使用ReentrantReadWriteLock。
4、限制并发的数量案例
import java.util.Random;
import java.util.concurrent.Semaphore;
import java.util.concurrent.TimeUnit;
/**
* @author xxx
* @descpription: 6辆车停 3个停车位
* Semaphore 维护了一个许可集,在许可可用前,会阻塞每一个 acquire(),然后再获取该许可
* @date 2024/7/17
*/
public class SemaphoreDemo {
public static void main(String[] args) {
//3个许可,最多3个线程同时访问;和线程池有 3 个线程同时访问
Semaphore semaphore = new Semaphore(3);
for (int i = 1; i <= 6; i++) {
new Thread(() -> {
try {
//获取许可
semaphore.acquire();
System.out.println(Thread.currentThread().getName() + " 抢到车位");
TimeUnit.MILLISECONDS.sleep(new Random().nextInt(10));
System.out.println(Thread.currentThread().getName() + " 离开车位");
} catch (InterruptedException e) {
throw new RuntimeException(e);
} finally {
//释放许可
semaphore.release();
}
}).start();
}
}
}