美文网首页
JDK 源码解析 —— Java 内存模型

JDK 源码解析 —— Java 内存模型

作者: 01_小小鱼_01 | 来源:发表于2018-05-25 19:52 被阅读79次

    Java 内存模型是围绕着在并发过程中如何处理原子性、 可见性和有序性来建立的。

    • 原子性:操作不可再分。 如在 Java 代码中使用 synchronized 和 ReentrantLock 来保障。
    • 可见性 A 线程操作, 对 B 线程可见, 既 A 变量赋值 1, B 线程能看见变量已经变为 1。 synchronized 、final、 volatile 和 ReentrantLock 都能提供。
    • 有序性:在本线程内观察, 所有操作都是有序的(线程内表现为串行语义); 如果在另一个线程中观察另一个线程, 所有操作都是无序的(指令重排序与工作内存与主内存同步延迟)。synchronized 、 volatile 和 ReentrantLock 都能提供有序性保证。

    从上述可以得出 synchronized 和 ReentrantLock 是万能的, 但是效率却有巨大的差别, ReentrantLock 会比 synchronized 好很多, ConcurrentHashMap 的源码实现中就利用了 ReentrantLock 分段锁来提升并发安全和效率。

    Happens-Before 规则

    要保证 A 线程看到 B线程的操作结果(无论 A 和 B 是否在同一线程中执行), 那么 A 和 B 之间必须满足 Happens-Before 关系。
    Happens-Before 规则如下:

    • 程序次序规则(Program Order Rule):程序中 操作 A 在操作 B 之前, 那么在线程中 A 操作将在 B 操作之前进行。

    • 管程锁定规则(Monitor Lock Rule):一个 unlock 操作 happen—before 后面(时间上的先后顺序,下同)对同一个锁的 lock 操作。

    • volatile 变量规则:对一个 volatile 变量的写操作 happen—before 后面对该变量的读操作。

    • 线程启动规则:Thread 对象的 start() 方法 happen—before 此线程的每一个动作。

    • 线程终止规则:线程的所有操作都 happen—before 对此线程的终止检测,可以通过 Thread.join() 方法结束、Thread.isAlive() 的返回值等手段检测到线程已经终止执行。

    • 线程中断规则:对线程 interrupt() 方法的调用 happen—before 发生于被中断线程的代码检测到中断时事件的发生。

    • 终结器规则(Finalizer Rule):一个对象的初始化完成(构造函数执行结束)happen—before 它的 finalize() 方法的开始。

    • 传递性:如果操作A happen—before 操作 B,操作 B happen—before 操作 C,那么可以得出 A happen—before 操作 C。

    如果两个操作间缺乏 Happens-Before 关系, 那么 JVM 可以对它们进行任意排序。

    更多内容
    深入理解Java内存模型(一)——基础

    相关文章

      网友评论

          本文标题:JDK 源码解析 —— Java 内存模型

          本文链接:https://www.haomeiwen.com/subject/qmnejftx.html