volatile在Java内存模型(JMM)中,保证共享变量对所有线程可见,但不保证原子性。volatile语义是同步,通过共享变量的方式,完成线程间的通信。

为什么需要volatile

Java内存模型中抽象、简化了计算机物理设备,分成工作内存和主内存,线程有各自的工作内存,却共享主内存。如果要把Java内存模型与物理设备映射起来的话,L1,L2 Cache可以视为工作内存,而L3 Cache视为主内存。线程执行指令时,会优先选择距离 CPU 较近的位置的工作内存中使用,而不会从读写速度较慢的主内存中,我称之为“就近原则”。当线程指令执行完后,赋值给工作内存,如果不回写到主内存,或者通知其他线程,其他线程是无法知晓变量已经修改,仍然会使用曾经缓存在工作内存中的变量,这就造成了缓存不一致的问题,Java使用volatile解决这种问题。volatile保证指令赋值完后的变量立即同步回主内存中,声明并通知其他线程当前赋值的变量已经失效,其他线程在下次使用时会放弃工作内存中变量,使用主内存中的变量。这样就完成了线程间对于volatile修饰的变量的通信。

可见性

执行引擎只与工作内存交互,再有工作内存与主内存交互。站在执行引擎的角度,与工作内存操作完成即表示指令执行完,但是什么时候工作内存会将结果刷新回主内存却不可预测。Java线程间的通信是通过共享内存的方式,线程A如果想通知其他所有线程(线程B,线程C)对于变量f的变化情况,需要满足两点:

  1. 将变量回写到主内存中
  2. 执行引擎读取时强制从主内存中加载
    在增加了增加了L1、L2 Cache之后,CPU何时将变量从独享缓存刷新会共享内存,独享缓存是否从共享内存加载变量,时间上都是不可确定的,这就造成了缓存不一致的问题。

可见性的语义是线程对变量更新操作后,其他线程是可以获知变量的变更情况。
工作内存和主内存关系

原子性

原子操作是一个或多个不可中断的操作,要么一次性完全执行完毕,要么就不执行,最终状态不存在有些操作执行完,有些操作没有执行,在外部看来是不可分割的整体(比如化学中的原子,当然原子也是可以再分割的,不过站在分子层面,原子是最小的不可分割的),原子操作关注的是不被线程调度器中断的操作。

原子性操作是不会出现线程交替执行的情况,如果出现线程交替,则说明操作被线程调度器中断。在Java内存模型中,原子性保证你获取的变量要么是初始值,要么是被某一个线程写入的值,而不会是有多个线程同一时间写入而产生的混乱结果,long或double类型除外(因为变量的前32位可能由一个线程写入,后32位由另一个不同的线程写入),不过加上volatile修饰后的long 和double也具有原子性。

注意:volatile关注可见性,而与原子性没有关系。volatile关注点在于从工作内存刷新回主内存,而原子操作关注的是否不被打断。原子和同步目的都是让不同线程可以安全地访问共享变量的两种处理方式,避免造成内存一致性错误。

我是葛一凡,希望对你有帮助。

参考

  1. 聊聊并发(一)深入分析Volatile的实现原理
  2. 聊聊并发(五)——原子操作的实现原理
  3. Java Language Specification
  4. Java Volatile 关键字详解
  5. 从缓存行出发理解volatile变量、伪共享False sharing、disruptor
  6. Java并发编程:volatile关键字解析
  7. Java 编程要点之并发(Concurrency)详解
  8. Java 并发编程(1): Java 内存模型(JMM)

volatile语义的更多相关文章

  1. 轻量级的同步机制——volatile语义详解(可见性保证+禁止指令重排)

    目录 1.关于volatile 2.语义一:内存可见性 2.1 一个例子 2.2 java的内存模型(JMM) 2.3 happens-before规则 2.4 volatile解决内存可见性问题的原 ...

  2. JAVA锁和volatile的内存语义&volatile的使用场景

    JAVA锁的内存语义 当线程释放锁时,JMM(Java Memory Model)会把该线程对应的本地内存中的共享变量刷新到主内存中. 当线程获取锁时,JMM会将该线程对应的本地内存置为无效.从而使得 ...

  3. Java内存模型-volatile的内存语义

    一 引言 听说在Java 5之前volatile关键字备受争议,所以本文也不讨论1.5版本之前的volatile.本文主要针对1.5后即JSR-133针对volatile做了强化后的了解. 二 vol ...

  4. Java并发编程原理与实战四十二:锁与volatile的内存语义

    锁与volatile的内存语义 1.锁的内存语义 2.volatile内存语义 3.synchronized内存语义 4.Lock与synchronized的区别 5.ReentrantLock源码实 ...

  5. java关键字volatile内存语义详细分析

    volatile变量自身具有下列特性. 1.可见性.对一个volatile变量的读,总是能看到(任意线程)对这个volatile变量最后的写 入. · 2.原子性:对任意单个volatile变量的读/ ...

  6. Java的多线程机制系列:不得不提的volatile及指令重排序(happen-before)

    一.不得不提的volatile volatile是个很老的关键字,几乎伴随着JDK的诞生而诞生,我们都知道这个关键字,但又不太清楚什么时候会使用它:我们在JDK及开源框架中随处可见这个关键字,但并发专 ...

  7. java并发编程(五)正确使用volatile

    转载请注明出处:     volatile用处说明     在JDK1.2之前,Java的内存模型实现总是从主存(即共享内存)读取变量,是不需要进行特别的注意的.而随着JVM的成熟和优化,现在在多线程 ...

  8. Java 理论与实践: 正确使用 Volatile 变量--转

    原文地址:http://www.ibm.com/developerworks/cn/java/j-jtp06197.html Java 语言中的 volatile 变量可以被看作是一种 “程度较轻的  ...

  9. Java 理论与实践: 正确使用 Volatile 变量

    Java 语言中的 volatile 变量可以被看作是一种 "程度较轻的 synchronized":与 synchronized 块相比,volatile 变量所需的编码较少,并 ...

随机推荐

  1. linux下bwa和samtools的安装与使用

    bwa的安装流程安装本软体总共需要完成以下两个软体的安装工作:1) BWA2) Samtools 1.BWA的安装a.下载BWA (download from BWA Source Forge ) h ...

  2. 一个专为电商定制的域名.shop

    2.73亿元人民币获得.shop域名的经营权,使shop域名成为最高节拍价的顶级域名.虽然最终“最高节拍价”被web域名打破,但在电商届域名里shop还是王者.shop作为一个主要面向线上.线下销售实 ...

  3. Linux嵌入式 -- 内核 - proc文件系统

    1. 什么是proc文件系统? 实例:通过 /proc/meminfo,查询当前内存使用情况. 结论:proc文件系统是一种在用户态检查内核状态的机制. 2.Proc文件分类 特点  每个文件都规定了 ...

  4. Oracle通过PLSQL进行数据表之间的同步

    昨天被要求拉取第三方oracle中的一个表数据,起初以为要导出表数据,然后再自己库中建个相同的表,然后导入数据,查过资料之后oracle可以通过dblink的方式同步表数据. 1.首先利用PLSQL工 ...

  5. 第一个Python程序hello.py提示出现File "<stdin>",line 1错误

    写第一个Python程序hello.py,内容仅有一句,print 'hello world', 运行 Python hello.py 出错,提示: File "<stdin>& ...

  6. 表格表格中获取不到button选择器

    今天做一个表单提交,怎么也拿不到button的选择器,不管用$(“#btn_update”)还会getElementById("btn_update"),浏览器也是谷歌没问题,后来 ...

  7. MRC与ARC混合编程的编译器标记

    如果是MRC项目创建ARC源文件,给这个源文件加上 -fobjc-arc 的编译器标记, 如果是ARC项目创建MRC源文件,给这个源文件加上 -fno-objc-arc 的编译器标记. 步骤: 1. ...

  8. 51nod 1449 贪心

    http://www.51nod.com/onlineJudge/questionCode.html#!problemId=1449 1449 砝码称重 题目来源: CodeForces 基准时间限制 ...

  9. 经典SQL语句(转载)

    一.基础 1.说明:创建数据库 CREATE DATABASE database-name 2.说明:删除数据库 drop database dbname 3.说明:备份sql server --- ...

  10. C# 之二进制序列化

    序列化:又称串行化,是.NET运行时环境用来支持用户定义类型的流化的机制.其目的是以某种存储形成使自定义对象持久化,或者将这种对象从一个地方传输到另一个地方. 一般有三种方式:1.是使用BinaryF ...