信号量(Semaphore)
信号量(Semaphore):信号量可以用来控制同时访问某个特定资源的线程数量,或者同时执行某个指定操作的数量(例:可比作是控制流量的红绿灯,比如XX马路要限制流量,只允许同时有一百辆车在这条路上行使,其他的都必须在路口等待,所以前一百辆车会看到绿灯,可以开进这条马路,后面的车会看到红灯,不能驶入XX马路,但是如果前一百辆中有五辆车已经离开了XX马路,那么后面就允许有5辆车驶入马路,这个例子里说的车就是线程,驶入马路就表示线程在执行,离开马路就表示线程执行完成,看见红灯就表示线程被阻塞,不能执行 *)。
Semaphore中管理着一组虚拟的许可(permit),许可的数量可通过构造方法来提定,在执行操作进先通过acquire方法获得许可(只要还有剩余的许可),并在使用后释放。如果没有许可,那么acquire方法将阻塞该线程直到有许可(或被中断或操作超时)。release方法可释放一个许可。
(JCIP5-14加上注释和main()):
/** * BoundedHashSet Using Semaphore to bound a collection * * @author Brian Goetz and Tim Peierls * @author Modified by alchimie */ public class BoundedHashSet<T> { private final Set<T> set; private final Semaphore sem; public BoundedHashSet(int bound) { this.set = Collections.synchronizedSet(new HashSet<T>()); // 设置许可数 sem = new Semaphore(bound); } // 获得一个许可并添加一个元素到集合中,如果失败则释放一个许可 public boolean add(T o) throws InterruptedException { sem.acquire(); boolean wasAdded = false; try { wasAdded = set.add(o); return wasAdded; } finally { if (!wasAdded) sem.release(); } } // 从集合中移除一个指定元素,如果成功就释放一个许可 public boolean remove(Object o) { boolean wasRemoved = set.remove(o); if (wasRemoved) sem.release(); return wasRemoved; } public void list() { for (T t : set) { System.out.println(t); } } public static void main(String args[]) throws InterruptedException { final BoundedHashSet<String> bhs = new BoundedHashSet<String>(3); new Thread(new Runnable() { public void run() { try { System.out.println(Thread.currentThread().getName() + "执行开始"); bhs.add("a"); bhs.add("b"); bhs.add("c"); System.out.println(Thread.currentThread().getName() + "执行完成"); } catch (InterruptedException e) { // TODO Auto-generated catch block e.printStackTrace(); } } }).start(); new Thread(new Runnable() { public void run() { try { System.out.println(Thread.currentThread().getName() + "执行开始"); bhs.add("d"); System.out.println(Thread.currentThread().getName() + "执行完成"); } catch (InterruptedException e) { // TODO Auto-generated catch block e.printStackTrace(); } } }).start(); bhs.list(); bhs.remove("c"); Thread.sleep(500); bhs.list(); } }
执行结果:
Thread-0执行开始
Thread-0执行完成
Thread-1执行开始
b
c
a
Thread-1执行完成
d
b
a
可以看到Thread-0将许可消耗完后,Thread-1将被阻塞,直到bhs.remove("c")执行后释放了一个许可才继续执行。
* 参考http://ifeve.com/tag/semaphore/
相关推荐
这本书的读书笔记涵盖了多个关键知识点,旨在帮助读者深入理解Java并发编程的核心概念。 1. **线程和进程的区别** - **线程** 是程序执行的最小单位,一个进程中可以有多个线程同时执行,共享同一块内存空间,通信...
《实战Java高并发程序设计》是一本专注于Java并发编程实践的书籍,试读版提供了前两章的内容,为读者提供了一个初步了解并发编程基础的窗口。在Java领域,并发编程是构建高性能、高效率系统的关键技术,对于软件开发...
Java并发编程是Java开发中的重要领域,特别是在大型分布式系统或者高并发应用中,对线程安全和性能优化的理解与实践至关重要。"JUC并发编程学习笔记(硅谷)"很可能包含了关于Java并发工具集(Java Util Concurrency, ...
Java并发编程是编程领域中的重要组成部分,特别是在大型系统和服务器端开发中不可或缺。Java自诞生以来就内置了对多线程的支持,使得开发者能够轻松创建并行运行的任务,提升程序性能。然而,随着并发编程实践的深入...
这篇“Java线程编程学习笔记(二)”很可能是对Java并发编程深入探讨的一部分,特别是涉及多线程示例的实践应用。我们将从标题、描述以及标签来推测可能涵盖的知识点,并结合"Multi-Threads Demo"这一压缩包文件名来...
《IT学习资料》-SSM实战项目-Java高并发秒杀API是一份全面的教程,旨在帮助学习者掌握如何在Java环境下使用Spring、SpringMVC和MyBatis(简称SSM)框架构建高并发的秒杀系统。这个项目涵盖了从项目初始化到实现完整...
《juc尚硅谷-自学笔记》是一份针对Java并发编程领域的深入学习资料,主要涵盖了Java Concurrency Utility(juc)包中的核心概念和技术。在这个压缩包中,"尚硅谷juc"很可能包含了视频教程、课件或者讲义,旨在帮助...
- Java支持原生多线程,简化并发编程。 - **线程的管理**: - 线程调度策略。 - **线程的调度**: - 操作系统的线程调度机制。 - **信号标志**: - 用于保护共享资源。 - **死锁及其避免**: - 通过合理的锁策略...
同时,理解并发编程原理,如锁、信号量和线程池,也是高级Java开发者必备的知识。 6. **职业感悟**:除了技术层面,这份指南还涵盖了职业发展的思考,包括如何进行有效学习、项目管理、团队协作和沟通技巧。这些软...
1. Java:异常处理、面向对象特性(封装、继承、多态)、集合框架(ArrayList、LinkedList、HashMap等)、并发编程(synchronized、volatile、ThreadLocal等)。 2. C/C++:指针、内存管理、预处理、模板、STL容器、...