volatile简介

java语言提供了一种稍弱的内存同步机制,即volatile变量。用来确保将变量的更新操作通知到其它线程,保证了新值能立即同步到主内存,以及每次使用前立即从内存刷新。当变量声明为volatile类型后,编译器与运行时都会注意到这个变量是共享的。

volatile变量对所有线程可见,对volatile变量所有的写操作都能立即反应到其它的线程中,但是volatile并不能保证操作的原子性,因此volatile变量的运算并不是线程安全的。

并发编程中的三个概念

在并发编程中,我们通常会遇到以下三个问题:原子问题,可见性问题,有序性问题。

原子性:

即一个操作或多个操作要么全部执行并且执行过程中不能被任何操作打断,要么都不执行。只有简单的读取、赋值(而且必须是将数字赋值给某个变量,变量 之间的赋值不是原子操作)才是原子操作。java内存模型只保证了基本读取和赋值是原子操作,如果要实现更大范围操作的原子性,可以通过 synchronized和lock来实现。由于synchronized和lock能保证任一时刻只有一个线程执行该程序块,那么自然就不存在原子性问 题了,从而保证了原子性。

可见性:

可见性是指多个线程访问同一个变量时,一个线程修改了这个变量的值,其它线程能够立即看到修改的值。对于可见性,java提供了volatile关 键字来保证可见性。当一个共享变量被volatile修饰的时候,它会保证修改的值立即被更新到内存,当有其它线程需要读取时,它会去内存中去读新值。而 普通共享变量不能保证可见性。因为普通共享变量被修改后,什么时候被写入内存是不确定的。当其他线程去读取新值时,此时内存中可能还是原本的值,因此无法 保证可见性。

有序性:

即程序执行的顺序按照代码的先后顺序执行。在java内存模型中,允许编译器和处理器对执行进行重排,但是重排过程不会影响单线程程序的执行,却会 影响到多线程并发执行的正确性。在java中,可以通过volatile关键字来保证一定的有序性。另外可以通过synchronized和lock来保 证有序性。很显然,synchronized和lock保证了每个时刻只有一个线程执行同步代码,相当于是让线程执行同步代码,自然保证了有序性。

指令重排:

一般来说,处理器为了提高程序运行效率,可能会对输入代码进行优化,它不保证程序中各个语句的执行顺序同代码中的顺序一致,但是它会保证程序最终执 行结果和代码顺序执行的结果是一致的。如果两个语句之间没有数据依赖,那么可能会被重排。指令重排不会影响单个线程的执行,但是会影响到线程并发执行的正 确性。

在并发编程中,如果想要一个程序正确的执行,必须保证原子性、可见性以及有序性,只要有一个没有被保证,就有可能会导致程序运行不正确。

volatile关键字的两层含义

一旦一个共享变量(类的成员变量、类的静态成员变量)被volatile修饰之后,那么就具备了两层含义:

1)保证了不同线程对这个变量进行操作时的可见性,即一个线程修改了某个变量的值,这个新值对其它线程来说是立即可见的。

2)禁止指令重排。

volatile关键字能够保证可见性,但是不能保证原子性。

volatile能禁止指令重排,所以volatile能在一定程度上保证有序性。

volatile关键字机制指令重排有两层意思:

1.当程序执行到volatile变量的读操作或者写操作时,在其前面的操作的更改肯定全部已经执行结束,并且结果已经对后面的修改可见,其后的操作一定还没有进行。

2.在进行指令优化时,不会对volatile变量进行重排。

从volatile关键字生成的汇编代码发现,加入volatile关键字时,会多出一个lock前缀指令。

lock前缀指令实际上相当于一个内存屏障。内存屏障会提供三个功能:

1)它确保指令重排时不会把其后面的指令排到内存屏障之前的位置,也不会把前面的指令排到内存屏障之后,即在执行到内存屏障这句指令时,在它前面的操作已经全部完成。

2)它会强制对缓存的修改操作立即写入内存。

3)如果是写操作,它会导致其他cpu中对应的缓存行无效。

使用volatile关键字的场景

synchronized关键字是防止多个线程同时执行一段代码,那么就会影响到程序的执行效率,而volatile关键字在某些情况下性能要优于 synchronized,但是要注意volatile关键字无法保证操作的原子性。通常来说,使用volatile必须具备以下两个条件。

1)对变量的写操作不依赖于当前值。

2)该变量没有包含在具有其它变量的不变式中。

意思就是保证操作是原子操作,才能保证使用volatile关键字的程序在并发时能够正确执行。

volatile简介的更多相关文章

  1. synchronized和volatile简介

    简介 volatile是一个变量修饰符,而synchronized是一个方法或块的修饰符.所以我们使用这两种关键字来指定三种简单的存取变量的方式. 2345678 int i1;int geti1() ...

  2. Java关键字:transient,strictfp和volatile简介

    关键字:transient 使用对象:字段 介绍:transient说明一个属性是临时的,不会被序列化. 当对象进行序列化(Serializable)过程时候,有一些属性的状态是瞬时的,这样的对象是无 ...

  3. volatile解析

    转载:http://www.importnew.com/17394.html 一.volatile简介: 在多线程并发编程中synchronized和Volatile都扮演着重要的角色,Volatil ...

  4. 5.彻底理解volatile

    1. volatile简介 在上一篇文章中我们深入理解了java关键字synchronized,我们知道在java中还有一大神器就是关键volatile,可以说是和synchronized各领风骚,其 ...

  5. Java并发编程,3分分钟深入分析volatile的实现原理

    volatile原理 volatile简介 Java内存模型告诉我们,各个线程会将共享变量从主内存中拷贝到工作内存,然后执行引擎会基于工作内存中的数据进行操作处理. 线程在工作内存进行操作后何时会写到 ...

  6. 彻底理解volatile,领悟其中奥妙

    本人免费整理了Java高级资料,涵盖了Java.Redis.MongoDB.MySQL.Zookeeper.Spring Cloud.Dubbo高并发分布式等教程,一共30G,需要自己领取.传送门:h ...

  7. 让你彻底理解volatile,面试不再愁

    本人免费整理了Java高级资料,涵盖了Java.Redis.MongoDB.MySQL.Zookeeper.Spring Cloud.Dubbo高并发分布式等教程,一共30G,需要自己领取.传送门:h ...

  8. java中的关键字volatile

    1.volatile简介 volatile作为java中的关键词之一,用以声明变量的值可能随时会被别的线程修改,使用volatile修饰的变量会强制将修改的值立即写入主存,主存中值的更新会使缓存中的值 ...

  9. 【C# 线程】 volatile 关键字和Volatile类、Thread.VolatileRead|Thread.VolatileWrite 详细 完整

    overview 同步基元分为用户模式和内核模式 用户模式:Iterlocked.Exchange(互锁).SpinLocked(自旋锁).易变构造(volatile关键字.volatile类.Thr ...

随机推荐

  1. requirejs中 shim

    使用requireJS的shim参数,完成jquery插件的加载 时间 2014-10-31 13:59:10  CSDN博客 原文  http://blog.csdn.net/aitangyong/ ...

  2. Objective-C语言分类与协议

    分类(Category)允许向一个类文件中添加新的方法声明,它不需要使用子类机制,并且在类实现的文件中的同一个名字下定义这些方法.其语法举例如下: #import "ClassName.h& ...

  3. 关于input内容监听(change)

    js的话利用input的onchange事件 而jquery的话利用change()函数,代码如下: <!DOCTYPE html> <html> <head> & ...

  4. url结构说明

    就以下面这个URL为例,介绍下普通URL的各部分组成 http://www.aspxfans.com:8080/news/index.asp?boardID=5&ID=24618&pa ...

  5. MAC: Homebrew(代替yum)安装

    安装    ruby -e "$(curl -fsSL https://raw.github.com/Homebrew/homebrew/go/install)"    最新方式请 ...

  6. 腾讯数据总监:运营人员必须掌握的APP基础数据分析体系(没有比这篇更系统全面的)

    导读:在互联网企业,任何一个APP都要事先规划好数据体系,才允许上线运营,有了数据才可以更好的科学运营.因此本文将为大家介绍APP的基础数据指标体系.主要分为五个维度,包括用户规模与质量.参与度分析. ...

  7. 一段处理百分数的js代码

    function percent(s, e, i){ s = Number(s), isNaN(s) && (s = "0"); var n = "%&q ...

  8. mybatis代码生成(generator工具生成代码)

    generator工具生成代码 下载地址     http://pan.baidu.com/s/1bY8C0I

  9. Ajax实现原理详解

    Ajax:Asynchronous javascript and xml,实现了客户端与服务器进行数据交流过程.使用技术的好处是:不用页面刷新,并且在等待页面传输数据的同时可以进行其他操作. 这就是异 ...

  10. 【leetcode❤python】 190. Reverse Bits

    #-*- coding: UTF-8 -*- class Solution:    # @param n, an integer    # @return an integer    def reve ...