返回顶部
首页 > 资讯 > 精选 >Java内存模型volatile的内存语义是什么
  • 812
分享到

Java内存模型volatile的内存语义是什么

2023-06-25 13:06:18 812人浏览 八月长安
摘要

这篇文章主要介绍“Java内存模型volatile的内存语义是什么”,在日常操作中,相信很多人在Java内存模型volatile的内存语义是什么问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Java内存模型v

这篇文章主要介绍“Java内存模型volatile的内存语义是什么”,在日常操作中,相信很多人在Java内存模型volatile的内存语义是什么问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Java内存模型volatile的内存语义是什么”的疑惑有所帮助!接下来,请跟着小编一起来学习吧!

1、volatile的特性

理解volatile特性的一个好办法是把对volatile变量的单个读/写,看成是使用同一个对单个读/写操作做了同步。

代码示例:

package com.lizba.p1;public class VolatileFeatureExample {        volatile long v1 = 0l;        public void set(long l) {        v1 = l;    }        public void getAndIncrement() {        v1++;    }        public long get() {        return v1;    }}

假设有多个线程分别调用上面程序的3个方法,这个程序在语义上和下面程序等价。

package com.lizba.p1;public class SynFeatureExample {        long v1 = 0L;        public synchronized void set(long l) {        v1 = l;    }        public void getAndIncrement() {        long temp = get();        // v1加一        temp += 1L;        set(temp);    }        public synchronized long get() {        return v1;    }}

如上两个程序所示,一个volatile变量的单个读\写操作,与一个普通变量的读\写操作都是使用同一个锁来同步,它们之间的执行效果相同。

上述代码总结:

锁的happens-before规则保证释放锁和获取锁的两个线程之间的内存可见性,这意味着对一个volatile变量的读,总能看到(任意线程)对这个volatile变量最后的写入。
锁的语义决定了临界区代码的执行具有原子性。这意味着,即使是64位的long型和double型变量,只要它是volatile变量,对该变量的读/写就具有原子性。如果是多个volatile操作或类似于volatile++这种复合操作,这些操作整体上不具备原子性。

总结volatile特性:

  • 可见性。对一个volatile变量的读,总是能看到(任意线程)对这个volatile变量最后的写入。

  • 原子性。对任意volatile变量的读/写具有原子性,但类似volatile++这种复合操作不具有原子性。

2、volatile写-读建立的happens-before关系

  • 对于程序员来说,我们更加需要关注的是volatile对线程内存的可见性。

jdk1.5(jsR-133)开始,volatile变量的写-读可以实现线程之间的通信。从内存语义的角度来说,volatile的写-读与锁的释放-获取有相同的内存效果。

  • volatile的写和锁的释放有相同的内存语义

  • volatile的读和锁的获取有相同的内存语义

代码示例:

package com.lizba.p1;public class VolatileExample {    int a = 0;    volatile boolean flag = false;    public void writer() {        a = 1;                              // 1        flag = true;                        // 2    }    public void reader() {        if (flag) {                         // 3            int i = a;                      // 4            System.out.println(i);        }    }    }

假设线程A执行writer()方法之后,线程B执行reader()方法。根据happens-before规则,

这个过程建立的happens-before关系如下:

  • 根据程序次序规则,1 happens-before 23 happens-before 4

  • 根据volatile规则,2 happens-before 3

  • 根据happens-before的传递性规则,1 happens-before 4

图示上述happens-before关系:

Java内存模型volatile的内存语义是什么

总结:这里A线程写一个volatile变量后,B线程读同一个volatile变量。A线程在写volatile变量之前所有可见的共享变量,在B线程读同一个volatile变量后,将立即对B线程可见。

3、volatile写-读的内存语义

volatile写的内存语义

当写一个volatile变量时,JMM会把该线程对应的本地内存中的共享变量值刷新到主内存。

以上面的VolatileExample为例,假设A线程首先执行writer()方法,随后线程B执行reader()方法,初始时两个线程的本地内存中的flag和a都是初始状态。

A执行volatile写后,共享变量状态示意图。

Java内存模型volatile的内存语义是什么

线程A在写flag变量后,本地内存A中被线程A更新过的两个共享变量的值被刷新到主内存中,此时A的本地内存和主内存中的值是一致的。

volatile读的内存语义

当读一个volatile变量时,JMM会把该线程对应的本地内存置为无效。线程接下来将会从主内存中读取共享变量。

B执行volatile读后,共享变量的状态示意图:

Java内存模型volatile的内存语义是什么

在读flag变量后,本地内存B包含的值已经被置为无效。此时,线程B必须从主内存中重新读取共享变量。线程B的读取操作将导致本地内存B与主内存中的共享变量的值变为一致。

总结volatile的写和volatile读的内存语义

  • 线程A写一个volatile变量,实质上是线程A向接下来将要读这个volatile变量的某个线程发出了(其对共享变量所做修改的)消息。

  • 线程B读一个volatile变量,实质上是线程B接收了之前某个线程发出的(在写这个volatile变量之前对共享变量所做修改的)消息。

  • 线程A写一个volatile变量,随后线程B读这个volatile变量,这个过程实质上是线程A通过主内存向线程B发送消息。

4、volatile内存语义实现

程序的重排序分为编译器重排序和处理器重排序(我的前面的博文内容有写哈)。为了实现volatile内存语义,JMM会分别禁止这两种类型的重排序。

volatile重排序规则表:

是否能重排序第二个操作

第一个操作普通读/写volatile读volatile写
普通读/写

NO
volatile读NONONO
volatile写
NONO

上图举例:第一行最后一个单元格意思是,在程序中第一个操作为普通读/写时,如果第二个操作为volatile写,则编译器不能重排序。

总结上图:

  • 第二个操作是volatile写时,都不能重排序。确保volatile写之前的操作不会被编译器重排序到volatile之后

  • 第一个操作为volatile读时,都不能重排序。确保volatile读之后的操作不会被编译器重排序到volatile之前

  • 第一个操作为volatile写,第二个操作为volatile读时,不能重排序。

为了实现volatile的内存语义,编译器在生成字节码时,会在指令序列中插入内存屏障来禁止特定类型的处理器重排序。

JMM采取的是保守策略内存屏障插入策略,如下:

  • 在每个volatile写操作屏障前面插入一个StoreStore屏障。

  • 在每个volatile写操作的后面插入一个StoreLoad屏障

  • 在每个volatile读操作的后面插入一个LoadLoad屏障。

  • 在每个volatile读操作的后面插入一个LoadStore屏障。

保守策略可以保证在任意处理器平台上,任意程序中都能得到正确的volatile内存语义。

保守策略下,volatile写插入内存屏障后生成的指令序列图:

Java内存模型volatile的内存语义是什么

解释:

StoreStore屏障可以保证在volatile写之前,其前面所有普通写操作已经对任意处理器可见了。这是因为StoreStore屏障将保障上面所有普通写在volatile写之前刷新到主内存。

保守策略下,volatile读插入内存屏障后生成的指令序列图:

Java内存模型volatile的内存语义是什么

解释:

LoadLoad屏障用来禁止处理器把上面的volatile读与下面的普通读重排序。LoadStore屏障用来禁止处理器把上面的volatile读与下面的普通写重排序。

上述volatile写和volatile读的内存屏障插入策略非常保守。在实际执行时,只要不改变volatile写-读的内存语义,编译器可以根据具体情况省略不必要的屏障。

代码示例:

package com.lizba.p1;public class VolatileBarrierExample {    int a;    volatile int v1 = 1;    volatile int v2 = 2;    void readAndWrite() {        // 第一个volatile读        int i = v1;        // 第二个volatile读        int j = v2;        // 普通写        a = i + j;        // 第一个volatile写        v1 = i + 1;        // 第二个volatile写        v2 = j * 2;    }    // ... 其他方法}

针对VolatileBarrierExamplereadAndWrite(),编译器生成字节码时可以做如下优化:

Java内存模型volatile的内存语义是什么

注意:最后的StoreLoad屏障无法省略。因为第二个volatile写之后,程序return。此时编译器无法准确断定后面是否会有volatile读写操作,为了安全起见,编译器通常会在这里插入一个StoreLoad屏障。

上面的优化可以针对任意处理器平台,但是由于不同的处理器有不同的“松紧度”的处理器内存模型,内存屏障的插入还可以根据具体的处理器内存模型继续优化。

X86处理器平台优化

X86处理器仅会对写-读操作做重排序。X86不会对读-读、读-写和写-写重排序,因此X86处理器会省略掉这3种操作类型对应的内存屏障。在X86平台中,JMM仅需要在volatile写后插入一个StoreLoad屏障即可正确实现volatile写-读内存语义。同时这样意味着X86处理器中,volatile写的开销会远远大于读的开销。

Java内存模型volatile的内存语义是什么

5、volatile和锁的比较

功能上:

锁比volatile更强大

可伸缩性和执行性能上:

volatile更具有优势

到此,关于“Java内存模型volatile的内存语义是什么”的学习就结束了,希望能够解决大家的疑惑。理论与实践的搭配能更好的帮助大家学习,快去试试吧!若想继续学习更多相关知识,请继续关注编程网网站,小编会继续努力为大家带来更多实用的文章!

--结束END--

本文标题: Java内存模型volatile的内存语义是什么

本文链接: https://lsjlt.com/news/304808.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • Java内存模型volatile的内存语义是什么
    这篇文章主要介绍“Java内存模型volatile的内存语义是什么”,在日常操作中,相信很多人在Java内存模型volatile的内存语义是什么问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Java内存模型v...
    99+
    2023-06-25
  • 并发编程之Java内存模型volatile的内存语义
    1、volatile的特性 理解volatile特性的一个好办法是把对volatile变量的单个读/写,看成是使用同一个锁对单个读/写操作做了同步。 代码示例: package ...
    99+
    2024-04-02
  • Java内存模型final的内存语义
    目录1、final域的重排序规则final2、写final域的重排序规则3、读final与的重排序规则4、final域为引用类型5、为什么final引用不能从构造函数内“逸出”6、f...
    99+
    2024-04-02
  • Java内存模型JMM与volatile
    目录1.Java内存模型2.并发三大特性2.1.原子性2.2.可见性2.3.有序性3.两个规则3.1.happens-before规则3.2.as-if-serial4.volati...
    99+
    2024-04-02
  • java高并发的volatile与Java内存模型是什么
    这篇文章将为大家详细讲解有关java高并发的volatile与Java内存模型是什么,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。public class Demo09 {&...
    99+
    2023-06-25
  • Java的内存模型是什么
    这篇文章主要介绍“Java的内存模型是什么”,在日常操作中,相信很多人在Java的内存模型是什么问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Java的内存模型是什么”的疑惑有所帮助!接下来,请跟着小编一起来...
    99+
    2023-06-15
  • 并发编程之Java内存模型锁的内存语义
    目录1、锁的释放-获取建立的happens-before关系2、锁释放和获取的内存语义3、锁内存的语义实现4、concurrent包的实现简介: 锁的作用是让临界区互斥执行。本文阐述...
    99+
    2024-04-02
  • 深入了解volatile和Java内存模型
    目录前言为什么我们需要volatile?保证数据的可见性禁止指令重排序Java内存模型(JMM)JMM下的内存逻辑结构内存交互的操作重排序Volatile实现原理禁止重排序实现原理可...
    99+
    2022-11-13
    Java内存模型 Java volatile Java 内存模型 volatile
  • 深度理解Java中volatile的内存语义
    volatile可见性实验 举个栗子 我这里开了两个线程,后面的线程去修改volatile变量,前面的线程不断获取volatile变量, 结果是会一致卡在死循环,控制台没有任何输出...
    99+
    2024-04-02
  • java高并发的volatile与Java内存模型详解
    public class Demo09 { public static boolean flag = true; public static class T1...
    99+
    2024-04-02
  • Java内存模型原理是什么
    这篇文章主要介绍“Java内存模型原理是什么”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“Java内存模型原理是什么”文章能帮助大家解决问题。内部原理JVM 中试图定义一种 JMM 来屏蔽各种硬件和...
    99+
    2023-06-02
  • Java内存模型的原理是什么
    这篇文章将为大家详细讲解有关Java内存模型的原理是什么,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。所有的编程语言中都有内存模型这个概念,区别于微架构的内存模型,高级语言的内存模型包括了编...
    99+
    2023-06-17
  • Java内存模型的规定是什么
    本篇内容介绍了“Java内存模型的规定是什么”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!首先要明确内存模型指什么。书中的定义是:在特定的操...
    99+
    2023-06-02
  • Java内存模型的概念是什么
    这篇文章主要介绍了Java内存模型的概念是什么的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇Java内存模型的概念是什么文章都会有所收获,下面我们一起来看看吧。内存模型,英文名Memory Model,他是一个...
    99+
    2023-06-02
  • 什么是JVM内存模型?
    本篇文章带大家初步了解一下JVM内存模型,有一定的参考价值,有需要的朋友可以参考一下,希望对大家有所帮助。计算机内存模型在程序运行时,CPU通过访问主存获取数据,但随着CPU的快速发展,CPU访问速度越来越高,硬件无法满足CPU的条件下,大...
    99+
    2016-08-29
    java教程 java JVM
  • Java8内存模型是什么
    这篇文章主要介绍了Java8内存模型是什么,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。一、JVM 内存模型根据 JVM 规范,JVM 内存共分为虚拟机栈、堆、方法区、程序计...
    99+
    2023-06-19
  • JMM内存模型是什么
    这篇文章主要介绍“JMM内存模型是什么”,在日常操作中,相信很多人在JMM内存模型是什么问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”JMM内存模型是什么”的疑惑有所帮助!接下来,请跟着小编一起来学习吧!JM...
    99+
    2023-06-27
  • golang内存模型是什么
    Golang的内存模型是一种并发内存模型,它定义了在不同Goroutine之间共享数据的行为。在Golang的内存模型中,每个Gor...
    99+
    2023-10-20
    golang
  • Redis的内存模型是什么
    Redis的内存模型是什么?针对这个问题,这篇文章给出了相对应的分析和解答,希望能帮助更多想解决这个问题的朋友找到更加简单易行的办法。redis有五种对象类型:字符串、哈希、列表、集合、有序集合,丰富的类型...
    99+
    2024-04-02
  • java中final域的内存语义是什么
    这篇文章将为大家详细讲解有关java中final域的内存语义是什么,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。1、说明对于final域的内存语义,编译器和处理器要遵守两个重排序规则(内部实现也是使用内存...
    99+
    2023-06-15
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作