Java并发编程 - volatile关键解析

volatile关键字可以说是Java虚拟机提供的最轻量级的同步机制,但是它并不容易弯曲被正确、完整的理解。

volatile的特性

volatile关键字是Java虚拟机提供的最轻量级的同步机制。Java内存模型对volatile专门定义了一些特殊的访问规则。

当一个变量被volatile修饰后,它将具备两种特性:

1. 可见性

** 第一是保证此变量对所以线程的可见性** ,这里的"可见性"是指当一个线程修改了这个变量的值,新值对于其它线程来说是可以立即得知的。而普通变量不能做到这一点,普通变量的值在线程间传递均需要通过主内存来完成,例如,线程A修改一个普通的变量的值,然后向主内存进行回写,另外一条线程B在线程A回写完成了之后再从主内存进行读取操作,新变量值才会对线程B可见。

关于volatile变量的可见性,经常会被开发人员误解,认为以下描述成立:"volatile变量对所有线程是立即可见,对volatile变量所有的写操作都能立刻反映到其他线程之中,换句话说,volatile变量在各个线程中是一致的,所以基于volatile变量的运算在并发情况下是安全的"。这句话论据部分并没有错,但是其论据并不能得出"基于volatile变量的运算在并发情况下是安全的"这个结论。volatile变量在各个线程的工作内存中不存在一致性问题(在各个线程的工作内存中,volatile变量也可以存在不一致的情况,但由于每次使用之前都要先刷新,执行引擎看不到不一致的情况,因此可以认为不存在一致性问题),但是Java里面的运算并非是原子操作,导致volatile变量的运算在并发情况下一样是不安全的。

例如:

package com.bytebeats.codelab;

import java.util.concurrent.CountDownLatch;

/**
 * ${DESCRIPTION}
 *
 * @author Ricky Fung
 */
public class VolatileDemo {

    public static void main(String[] args) throws InterruptedException {

        VolatileDemo demo = new VolatileDemo();
        demo.run();
    }

    public void run() throws InterruptedException {

        int threadNum = 20;
        final CountDownLatch latch = new CountDownLatch(threadNum);
        for (int i=0; i<threadNum; i++){

            Thread t = new Thread(new Runnable() {
                @Override
                public void run() {

                    for(int x=0; x<50; x++){
                        incr();
                    }
                    latch.countDown();
                }
            });
            t.start();
        }

        //等待所有线程执行完累加操作
        latch.await();

        System.out.println(race);
    }

    private volatile int race = 0;

    private void incr(){
        race++;
    }
}

模拟了20个线程对race变量进行自增操作,如果这段代码是并发安全的话,最后输出的结果应该为1000。但事实上每次运行上述代码,输出的结果可能都会不一样,都是一个小于1000的数。

究其原因,就是Java 中 race++不是原子操作。

由于volatile变量,只能保证可见性,在不符合以下两条规则的情况下,我们仍然需要通过加锁(使用synchronized 或 java.util.concurrent中的原子类)来保证原子性。

  • 运算结果并不依赖变量的当前值,或者能够确保只有单一的线程修改变量的值。
  • 变量不需要与其他的状态变量共同参与不变约束。

2. 禁止指令重排序优化

使用volatile变量的第二个语义是禁止指令重排序优化,

原子性、可见性、有序性

Java内存模型是围绕着在并发过程中如何处理 原子性、可见性和有序性 这3个特征来建立的。我们来逐个看一下哪些操作实现了这3个特性。

原子性

由Java内存模型来直接保证的原子性变量操作包括:read、load、assign、use、store和write,我们大致可以认为基本数据类型的访问操作是具备原子性的(例外的是long和double的非原子性协定)。

如果硬要程序需要一个更大范围的原子性保证(经常会遇到),Java内存模型还提供了synchronized 关键字和Lock,在synchronized 块之间的操作也具备原子性。

可见性

可见性是指当一个线程修改了这个变量的值,新值对于其它线程来说是可以立即得知这个修改。

Java内存模型是通过在变量修改后将新值同步回主内存,在变量读取前从主内存刷新变量值 这种依赖主内存作为传递媒介的方式来实现可见性的,无论是普通变量还是volatile变量都是如此,普通变量与volatile变量的区别是:volatile的特殊规则保证了新值能立即同步到主内存,以及每次使用前立即从主内存刷新。因此,可以说volatile保证了多线程操作时 变量的可见性,而普通变量则不能保证这一点。

除了volatile之外,Java还有两个关键字能实现可见性,即synchronized和final。

有序性

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

推荐阅读更多精彩内容