Java信号量Semaphore原理及代码实例
作者:cuisuqiang 时间:2021-10-02 21:55:08
Semaphore 通常用于限制可以访问某些资源(物理或逻辑的)的线程数目。自从5.0开始,jdk在java.util.concurrent包里提供了Semaphore 的官方实现,因此大家不需要自己去实现Semaphore。
下面的类使用信号量控制对内容池的访问:
import java.util.concurrent.Semaphore;
class Pool {
private static final int MAX_AVAILABLE = 100;
private final Semaphore available = new Semaphore(MAX_AVAILABLE, true);
public Object getItem() throws InterruptedException {
available.acquire(); // 从此信号量获取一个许可,在提供一个许可前一直将线程阻塞,否则线程被中断
return getNextAvailableItem();
}
public void putItem(Object x) {
if (markAsUnused(x))
available.release(); // 释放一个许可,将其返回给信号量
}
// 仅作示例参考,非真实数据
protected Object[] items = null;
protected boolean[] used = new boolean[MAX_AVAILABLE];
protected synchronized Object getNextAvailableItem() {
for (int i = 0; i < MAX_AVAILABLE; ++i) {
if (!used[i]) {
used[i] = true;
return items[i];
}
}
return null;
}
protected synchronized boolean markAsUnused(Object item) {
for (int i = 0; i < MAX_AVAILABLE; ++i) {
if (item == items[i]) {
if (used[i]) {
used[i] = false;
return true;
} else
return false;
}
}
return false;
}
}
虽然JDK已经提供了相关实现,但是还是很有必要去熟悉如何使用Semaphore及其背后的原理。
做一个简单的Semaphore实现:
class SemaphoreTest {
private boolean signal = false;
public synchronized void take() {
this.signal = true;
this.notify();
}
public synchronized void release() throws InterruptedException {
while (!this.signal)
wait();
this.signal = false;
}
}
使用这个semaphore可以避免错失某些信号通知。用take方法来代替notify,release方法来代替wait。如果某线程在调用release等待之前调用take方法,那么调用release方法的线程仍然知道take方法已经被某个线程调用过了,因为该Semaphore内部保存了take方法发出的信号。而wait和notify方法就没有这样的功能。
可计数的Semaphore:
class SemaphoreTest {
private int signals = 0;
public synchronized void take() {
this.signals++;
this.notify();
}
public synchronized void release() throws InterruptedException {
while (this.signals == 0)
wait();
this.signals--;
}
}
Semaphore上限:
class SemaphoreTest {
private int signals = 0;
private int bound = 0;
public SemaphoreTest(int upperBound) {
this.bound = upperBound;
}
public synchronized void take() throws InterruptedException {
while (this.signals == bound)
wait();
this.signals++;
this.notify();
}
public synchronized void release() throws InterruptedException {
while (this.signals == 0)
wait();
this.signals--;
this.notify();
}
}
当已经产生的信号数量达到了上限,take方法将阻塞新的信号产生请求,直到某个线程调用release方法后,被阻塞于take方法的线程才能传递自己的信号。
把Semaphore当锁来使用:
当信号量的数量上限是1时,Semaphore可以被当做锁来使用。通过take和release方法来保护关键区域。
来源:https://www.iteye.com/blog/cuisuqiang-2020146
标签:Java,信号量,Semaphore
0
投稿
猜你喜欢
C#的正则表达式Regex类使用简明教程
2021-11-16 12:47:32
C#面向对象编程之猜拳游戏实现方法
2021-11-17 23:25:52
Java调用.dll文件的方法
2023-11-23 21:16:22
springboot默认日志框架选择源码解析(推荐)
2021-06-19 08:02:51
android球形水波百分比控件代码
2021-06-20 06:03:34
JFileChooser实现对选定文件夹内图片自动播放和暂停播放实例代码
2021-10-02 15:41:18
Java 通过AQS实现数据组织
2023-04-05 22:19:29
Spring AOP实现功能权限校验功能的示例代码
2021-10-18 21:32:29
Java Hutool工具实现验证码生成及Excel文件的导入和导出
2023-02-04 22:49:32
SpringBoot中@ConfigurationProperties注解实现配置绑定的三种方法
2023-03-19 12:36:25
Android实现时钟特效
2022-09-08 01:43:22
Java 超详细讲解IO操作字节流与字符流
2022-04-28 20:09:57
C++实现的链表类实例
2023-07-04 08:36:36
Java接口和抽象类有什么区别
2021-08-21 07:01:18
java程序代码与文本对比实用工具简介
2023-07-18 22:57:05
macOS下Spring Boot开发环境搭建教程
2023-11-05 15:50:59
不看后悔!揭秘游戏服务器开发
2023-08-22 16:20:30
java实现一次性压缩多个文件到zip中的方法示例
2021-11-16 07:24:47
使用java编程从0到1实现一个简单计算器
2023-12-06 04:47:53
C#中的Lazy如何使用详解
2023-10-25 12:56:34