java.util.concurrent包是在并发中很常用的实用工具类。此包包括了几个小的、已标准化的可扩展框架,以及一些提供有用功能的类,没有这些类,这些功能很难实现或实现起来冗长乏味。下面简要描述主要的组件。
一、执行程序
1、接口
- Executor是一个简单的标准化接口,用于定义类似于线程的自定义子系统,包括线程池、异步I/O和轻量级任务框架。根据所使用的具体Executor类的不同,可能在新创建的线程中,现有的任务执行线程中,或者调用execute()的线程中执行任务,并且可能顺序或并发执行。
- ExecutorService提供了多个完整的异步任务执行框架。ExecutorService管理任务的排队和安排,并允许受控制的关闭。
- ScheduledExecutorService子接口添加了对延迟的和定期任务执行的支持。
- ExecutorService提供了安排异步执行的方法,可执行由Callable表示的任何函数,结果类似于Runnable。
- Future返回函数的结果,允许确定执行是否完成,并提供取消执行的方法。
2、实现类
- 类ThreadPoolExecutor和ScheduledThreadPoolExecutor提供可调的、灵活的线程池。
- Executors类提供大多数Executor的常见类型和配置的工厂方法,以及使用它们的集中实用工具方法。
- 其它基于Executor的实用工具包括具体类FutureTask,它提供Future的常见扩展实现,以及ExecutorCompletionService,它有助于协调对异步任务组的处理。
二、队列
- java.util.concurrent.ConcurrentLinkedQueue<E>提供了高效的、可伸缩的、线程安全的非阻塞FIFO队列
- java.util.concurrent中的五个实现都支持扩展的BlockingQueue接口,该接口定义了put和take的阻塞版本:LinkedBlockingQueue、ArrayBlockingQueue、SynchronousQueue、PriorityBlockingQueue和DelayQueue。这些不同的类覆盖了生产者-使用者、消息传递、并行任务执行和相关并发设计的大多数常见使用的上下文。
三、计时
TimeUnit类为指定和控制基于超时的操作提供了多重粒度(包括纳秒级)。该包中的大多数类除了包含不确定的等待之外,还包含基于超时的操作。在使用超时的所有情况中,超时指定了在表明已超时前该方法应该等待的最少时间。在超时发生后,实现会“尽力”检测超时。但是,在检测超时与超时之后再次实际执行线程之间可能要经过不确定的时间。
四、同步器
四个类可协助实现常见的专用同步语句
- Semaphore是一个经典的并发工具。
- CountDownLatch是一个极其简单但又极其常用的实用工具,用于在保持给定数目的信号、事件或条件前阻塞执行。
- CyclicBarrier是一个可重置的多路同步点,在某些并行编程风格中很有用。
- Exchanger允许两个线程在集合点交换对象,它在多流水线设计中是有用的。
五、并发Collection
除队列外,此包还提供了几个设计用于多线程上下文中的Collection实现:
- ConcurrentHashMap
- CopyOnWriteArrayList
- CopyOnWriteArraySet
此包中与某些类一起使用的“Concurrent”是一种简写,表明与类似的“同步”类有所不同。例如,java.util.Hashtable和Collections.synchronizeMap(new HashMap())是同步的,但ConcurrentHashMap则是并发的。并发集合是线程安全的,但是不受单个排它锁定的管理。在ConcurrentHashMap这一特定情况下,它可以安全的允许进行任意数目的并发读取,以及数目可调的并发写入。需要通过单个锁定阻止对集合的所有访问时,“同步”类是很有用的,其代价是较差的可伸缩性。在期望多个线程访问公共集合的其他情况中,通常“并发”版本要更好一些。当集合是未共享的,或者仅保持其他锁定时集合是可访问的情况下,非同步集合则要更好一些。
大多数并发Collection实现(包括大多数Queue)与常规的java.util约定也不同,因为它们的迭代器提供了“弱一致”的,而不是快速失败的遍历。“弱一致”的迭代器是线程安全的,但是在迭代时没有必要冻结集合,所以它不一定反应自迭代器创建以来的所有更新。
六、内存一致性属性
Java Language Specification 第17章定义了内存操作(如共享变量的读写)的“happen-before”的关系。只有写入操作happen-before读取操作时,才保证一个线程写入的结果对另一个线程的读取是可视的。synchronized和volatile构造happen-before的关系,Thread.start()和Thread.join()方法形成happen-before关系。尤其是:
- 线程中的每个操作happen-before稍后按线程顺序传入的该线程中的每个操作。
- 一个解除锁监视器的(synchronized阻塞或方法退出)happen-before相同监视器的每个后续锁(synchronized阻塞或方法进入)。并且因为happen-before关系是可传递的,所以解除锁定之前的线程的所有操作happen-before锁定该监视器的任何线程后续的所有操作。
- 写入volatile字段happen-before每个后续读取相同字段。volatile字段的读取和写入与进入和退出监视器具有相似的内存一致性效果,但不需要互斥锁。
- 在线程上调用start happen-before已启动的线程中的任何线程。
- 线程中的所有操作 happen-before从该线程上的join成功返回的任何其他线程。
java.util.concurrent中所有类的方法及其子包扩展了这些对更高级别同步的保证。尤其是:
- 线程中将一个对象放入任何并发collection之前的操作 happen-before 从另一个线程中的collection访问或移除该元素的后续操作。
- 线程中向Executor提交Runnable之前的操作 happen-before 其执行开始。同样适用于向ExecutorService提交Callables。
- 异步计算(由Future表示)所采取的操作 happen-before 通过另一线程中 Future.get() 获取结果后续的操作。
- “释放”同步存储方法(如 Lock.unlock、Semaphore.release 和 CountDownLatch.countDown())之前的操作 happen-before 另一线程中相同同步存储对象成功“获取”方法(如 Lock.lock、 Semaphore.acquire、 Condition.await 和 CountDownLatch.await)的后续操作。
- 对于通过Exchanger成功交换对象的每个线程对,每个线程中exchange()之前的操作 happen-before 另一个线程中对应 exchange() 后续的操作。
- 调用 CyclicBarrier.await 之前的操作 happen-before 屏障操作所执行的操作,屏障操作所执行的操作 happen-before 从另一个线程中对应 await 成功返回的后续操作。
备注:
happen-before(先行发生)规则: Java内存模型中定义的两项操作之间的偏序关系,如果操作A先行发生于操作B,其意思就是说,在发生操作B之前,操作A产生的影响都能被操作B观察到,“影响”包括修改了内存中共享变量的值、发送了消息、调用了方法等,它与时间上的先后发生基本没有基本没有太大关系。这个原则非常重要,它是判断数据数据是否存在竞争、线程是否安全的主要依据。
本文源自Java 2 Platform SE 5.0 API(即JDK 1.5/1.6 API)中,对于java.util.concurrent包的描述
本文是目前个人理解的结果,仅供参考,如后续发现问题,本人会进行相应的更正,也欢迎各位对Java线程感兴趣的朋友或者前辈进行指正。
相关推荐
Java.util.concurrent(JUC)是Java平台中的一个核心包,专门用于处理多线程并发问题。这个包包含了大量的工具类和接口,极大地简化了并发编程的复杂性,提高了程序的性能和可伸缩性。本测试源文件主要是针对JUC并发...
Java线程学习笔记涉及了Java多线程编程的多个关键知识点,本篇知识点整理将详细解释每个概念及其在Java中的实现方式。 基本知识部分包含了Java线程编程的基础内容,它们是并发编程的基石。 任务Runnable是一个接口...
- `java.util.concurrent.ThreadPoolExecutor`:普通线程池的核心实现,可配置核心线程数、最大线程数、线程空闲超时时间等参数。 - `java.util.concurrent.ScheduledThreadPoolExecutor`:调度线程池的核心实现,...
2. **线程同步**:为了解决多个线程间的竞态条件,Java提供了多种同步机制,如`synchronized`关键字、`Lock`接口(包括`ReentrantLock`)以及`java.util.concurrent`包中的各种工具类,如`Semaphore`、`...
10. **原子类**:`java.util.concurrent.atomic`包下的原子类如`AtomicInteger`、`AtomicLong`等,提供了基于CAS(Compare and Swap)的无锁操作,用于在多线程环境下实现高效且线程安全的操作。 11. **Future和...
8. **并发集合**:Java的并发包(java.util.concurrent)提供了线程安全的集合,如ConcurrentHashMap、CopyOnWriteArrayList等,它们在内部实现了高效的并发控制。 9. **线程中断**:通过Thread.interrupt()方法...
锁机制包括内置锁(也称为监视器锁,由synchronized实现)和显式锁(如java.util.concurrent.locks.Lock接口的实现,如ReentrantLock)。此外,还可以使用volatile关键字保证共享变量的可见性和有序性,但不能保证...
### Java分布式应用学习笔记05多线程下的并发同步器 #### 1. 前言 在现代软件开发中,特别是在分布式系统和高性能计算领域,有效地管理多线程之间的协同工作至关重要。Java语言提供了丰富的工具和API来帮助开发者...
4. lock锁:Lock接口是Java并发包(java.util.concurrent.locks)提供的高级锁,相比synchronized,它提供了更细粒度的锁控制,如可中断锁、可重入锁和公平锁。ReentrantLock是Lock的一个实现,它支持锁的获取和释放...
java.util.concurrent.Callable ; import java.util.concurrent.ExecutionException ; import java.util.concurrent.FutureTask ; /** * @Author IamZY * @create 2019/12/28 14:57 */ public class CallableDemo { ...
在这个学习笔记中,主要讨论了Java中的线程同步机制,包括volatile关键字、synchronized以及Lock接口,特别是ReentrantLock的使用。 首先,对于线程1和线程2的疑惑,调试(debug)模式并不能改变线程的执行顺序。...
Java的并发库`java.util.concurrent`提供了高级并发工具,如ExecutorService、Future、CountDownLatch等,方便开发者高效地进行并发编程。 **并发编程代码示例** 压缩包中的`并发编程代码.zip`包含了实际的并发编程...
Java多线程是Java编程中的重要概念,它允许程序同时执行多个任务,从而提高系统效率和资源利用率。在Java中,实现多线程有两种主要方式:通过实现`Runnable`接口和继承`Thread`类。 首先,让我们从创建线程开始。当...
可重入锁是Java.util.concurrent.locks包下的ReentrantLock类,支持公平锁和非公平锁,具有比synchronized更细粒度的控制。它提供tryLock()方法,可以在无法获取锁时立即返回,而不是等待。 3. **ThreadLocal**: ...
Java提供了多种工具,如`synchronized`关键字、`java.util.concurrent`包,来保证线程安全。 12. **Java的安全API**:如`java.security`包中的API,包括密钥和证书管理,数字签名,消息摘要,随机数生成等,这些都...
4. **并发集合**:Java并发包(`java.util.concurrent`)提供了一系列线程安全的集合,如`ConcurrentHashMap`, `CopyOnWriteArrayList`, `BlockingQueue`等。笔记会解析它们的设计原理和使用技巧。 5. **线程池**:`...
Java JDK 5.0是Java开发工具包的一个重要版本,由Sun Microsystems(后被Oracle收购)于2004年发布。...通过阅读“良葛格Java JDK 5.0学习笔记”,读者可以系统地学习和掌握这些内容,进一步提升自己的Java编程技能。
4. **并发编程增强**:新增了`java.util.concurrent`包中的工具类,如`ExecutorService`、`Future`等,简化了多线程编程。 5. **改进的垃圾收集**:引入了G1垃圾收集器,这是一种更高效的垃圾回收策略,适用于大内存...