Java volatile关键字

volatile是Java虚拟机提供的最轻量级的同步机制,用于保证共享变量在多线程之间的可见性。谈以下几点体会:

1. 可见性是如何保证的?

cpu访问内存中的数据,会先加载到自身的缓存,再从缓存中加载数据,处理后,会再次写入到缓存。为了优化cpu的处理能力,缓存中的数据,可能不会第一时间写入到主存,方便cpu对数据的频繁访问。

volatile关键字修饰的变量,会在cpu处理完成后,第一时间刷新到主存,保证其他线程能在第一时间读取到变量的变化。

刷新是如何实现的呐?

通过分析对volatile修饰的变量进行写操作的汇编代码,可以发现Lock前缀指令,该指令在多核处理器有两个作用:

  1. 将当前处理器缓存行的数据写回到系统内存(执行指令期间,可以独占内存,通过锁缓存或是总线)
  2. 这个写回操作会使其他cpu里缓存了该内存地址的数据无效,即读取该变量时需要从主存重新读取变量到缓存,给cpu使用(通过缓存一致性协议保证,相关内容后续补充)

2. volatile关键字并不一定保证并发安全

volatile关键字只能保证所有线程在同一时间,看到的变量值是唯一的。即只有对volatile修饰的变量进行原子性操作,才会保证一个线程写操作变量后,立即刷新到主存,其他线程会等候读取到。原子性操作比如:

public volatile int a = 0;
//原子操作
public void set(int input){
  a = input;
}

但如果针对volatile修饰的变量进行非原子性操作(例如i++,自增运算在class文件中,是由四条字节码组成的,底层可能由更多条机器指令组成,并非原子操作),其他线程就由可能在此期间访问到这个变量,导致变量在各线程中不一致,从而导致线程的不安全。这并不违背volatile变量的语义规范。示例如下:

import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

/**
 * Created by che on 2017/4/14.
 */
public class VolatileTest implements Runnable {
    private static volatile int in = 0;
    public static void increament(){
        in++;
    }
    public void run(){
        for(int i = 0; i < 1000; i++){
            increament();
        }
    }
    public static void main(String[] args) {
        ExecutorService executorService = Executors.newCachedThreadPool();
        for(int i = 0; i < 20; i++){
            executorService.execute(new VolatileTest());
        }
        executorService.shutdown();
        System.out.println(in);
    }
}

程序中想证明的是,用20个线程来做自增操作,每个线程自增1000次,如果并发安全,in的值应为20000,实际运行结果均小于20000。原因就是increament()操作并非原子操作,可以使用synchronized关键字来保证区块的原子性,即在上文中做这样的修改:

    public synchronized static void increament(){
        in++;
    }

3. volatile 对指令重排序的影响

重排序是指编译器和处理器为了优化程序性能而对指令进行的重新排序,通俗来讲,就是我们按顺序写的代码,编译器和处理器会进行优化,因而变的无序。虽然顺序变了,但在单线程的执行背景下,是会获得一致性的结果的。但在多线程的条件下,就可能产生错误。
volatile变量,会通过对其读/写操作时,分别加入不同的内存屏障,来影响重排序的发生,JMM的保守策略如下:

  • volatile写操作前插入StoreStore屏障
  • volatile写操作后插入StoreLoad屏障
  • volatile读操作后插入LoadLoad屏障
  • volatile读操作后插入LoadStore屏障

注:X86处理器仅会对写--读操作做重排序,不会对读--读、读--写、写--写操作重排序,因此只需要插入一个StoreLoad。这意味着在X86处理器中,volatile写的开销比volatile读的开销会大很多(因为执行StoreLoad屏障开销会比较大)。

4. volatile与synchronized的选择

volatile是轻量级的同步机制,相对而言,synchronized是一种锁的机制,对底层的JVM优化限制更多,性能会下降。如果并发编程经验不足或对并发场景认识不足,建议使用synchronized,毕竟安全远大于性能。
「此部分待后续补充」


Reference:
《深入理解Java虚拟机》
《Java并发编程艺术》

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,099评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,828评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,540评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,848评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,971评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,132评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,193评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,934评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,376评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,687评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,846评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,537评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,175评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,887评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,134评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,674评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,741评论 2 351

推荐阅读更多精彩内容