concurrent 包
由于 java 的 CAS 同时具有 volatile 读和 volatile 写的内存语义,因此 java 线程之间的通信现在有了下面四种方式:
-
A 线程写 volatile 变量,随后B线程读这个 volatile 变量。
-
A 线程写 volatile 变量,随后 B 线程用 CAS 更新这个 volatile 变量。
-
A 线程用 CAS 更新一个 volatile 变量,随后 B 线程用 CAS 更新这个 volatile 变量。
-
A 线程用 CAS 更新一个 volatile 变量,随后B线程读这个 volatile 变量。
ps: 简而言之,volatile 和 CAS 的读写组合。
Java 的 CAS 会使用现代处理器上提供的高效机器级别原子指令,这些原子指令以原子方式对内存执行读-改-写操作,这是在多处理器中实现同步的关键(从本质上来说,能够支持原子性读-改-写指令的计算机器,是顺序计算图灵机的异步等价机器,因此任何现代的多处理器都会去支持某种能对内存执行原子性读-改-写操作的原子指令)。
同时,volatile 变量的读/写和CAS可以实现线程之间的通信。
把这些特性整合在一起,就形成了整个 concurrent 包得以实现的基石。
通用模式
如果我们仔细分析 concurrent 包的源代码实现,会发现一个通用化的实现模式:
首先,声明共享变量为 volatile;
然后,使用 CAS 的原子条件更新来实现线程之间的同步;
同时,配合以 volatile 的读/写和CAS所具有的 volatile 读和写的内存语义来实现线程之间的通信。
AQS
AQS,非阻塞数据结构和原子变量类(java.util.concurrent.atomic
包中的类)
这些 concurrent 包中的基础类都是使用这种模式来实现的,而 concurrent 包中的高层类又是依赖于这些基础类来实现的。
参考资料
-
JSR 133
-
other
http://www.infoq.com/cn/articles/java-memory-model-5