Volatile的使用

volatile变量在Java中被看做是"程度较轻的synchronized",与synchronized相比,volatile变量的编码较少,运行时开销也小,所以它所实现的功能也只是synchronized中的一部分。

Java内存模型中可见性原子性有序性

  • 可见性
    线程之间的可见性,一个线程修改后的结果,另一个线程马上就能见到。比如:用volatile修饰的变量就具有可见性。volatile修饰的变量不允许内存缓存和重排序,是直接在内存中更改。volatile只能保证它所修饰内容的可见性,但是保证不了修饰内容的原子性。比如:volatile int a=1,这个操作之后是a+=1,这个变量具有可见性,但是a+=1它不是原子操作,也就是这个操作存在线程安全问题。
  • 原子性
    原子是世界上最小的单位,具有不可分割性。比如:a=0这个操作不可分割,所以这个操作是原子操作;在比如:a+=1,这个操作可以分割为a=a+1,所以它不是原子操作。
  • 有序性
    Java 语言提供了 volatilesynchronized 两个关键字来保证线程之间操作的有序性,volatile 是因为其本身包含“禁止指令重排序”的语义,synchronized 是由“一个变量在同一个时刻只允许一条线程对其进行 lock 操作”这条规则获得的,此规则决定了持有同一个对象锁的两个同步块只能串行执行。

根据上面的叙述,来分析下面的代码:

public class VolatileClass {
    private volatile int m = 0;

    public static void main(String[] args) throws InterruptedException {
        VolatileClass volatileClass = new VolatileClass();

       for (int i = 0; i < 4; i++) {

            Thread threadA = new Thread(volatileClass.new MyThread());
            threadA.start();

            Thread thread = new Thread(volatileClass.new MyThread());
            thread.start();

            threadA.join();
            thread.join();
            System.out.println("CurrThread:" + Thread.currentThread().getName() + ",m:" + volatileClass.m);
            volatileClass.m = 0;
        }
    }

    private class MyThread implements Runnable {

        @Override
        public void run() {
            for (int i = 0; i < 1000; i++) {
                inscribe();
            }
        }
    }

    private void inscribe() {
        m ++;
    }
}

程序运行之后,m的值最终为多少呢?2000?其实你多运行几次就会发现,它每次的值都不尽相同,但是都小于等于2000。看下面的运行结果:

CurrThread:main,m:1946
CurrThread:main,m:2000
CurrThread:main,m:2000
CurrThread:main,m:1754

将方法加上synchronized修饰,这样方法就具备了原子性,代码如下:

private synchronized void inscribe() {
        m += 1;
        System.out.println("CurrThread:" + Thread.currentThread().getName() + "-->m:" + m);
    }

运行程序:

CurrThread:main,m:2000
CurrThread:main,m:2000
CurrThread:main,m:2000
CurrThread:main,m:2000

看看JMM中内存与线程之间的关系?如下:

image.png

JMM中的内存分为主内存和工作内存,其中主内存是所有线程共享的,而工作内存是每个线程独立分配的,各个线程的工作内存之间相互独立、互不可见。在线程启动的时候,虚拟机为每个内存分配了一块工作内存,不仅包含了线程内部定义的局部变量,也包含了线程所需要的共享变量的副本,当然这是为了提高执行效率,读副本的比直接读主内存更快。

对于Volatile修饰的变量,当要获取值时,直接从内存中获取最新值;当要写入值时,也是直接写入内存,而不是从工作内存中。Volatile修饰的变量直接跳过了这一步。
当把变量声明为volatile类型后,编译器与运行时都会注意到这个变量是共享的,因此不会将该变量上的操作与其他内存操作一起重排序。volatile变量不会被缓存在寄存器或者对其他处理器不可见的地方,因此在读取volatile类型的变量时总会返回最新写入的值。
在访问volatile变量时不会执行加锁操作,因此也就不会使执行线程阻塞,因此volatile变量是一种比sychronized关键字更轻量级的同步机制。
当一个变量被定义为volatile时,就具备了两大特性:可见性有序性

volatile使用场景

要使用volatile必须具备以下两点:

  • 对变量的写操作不依赖于当前值
  • 该变量没有包含在具有其它变量的不变式中

事实上,上面的两个条件就是要保证volatile变量操作的原子性,这样才能使程序在并发的时候能够正确使用。
看如下代码:

public class VolatileClass {
    private boolean flag = false;

    public static void main(String[] args) throws InterruptedException {
        VolatileClass volatileClass = new VolatileClass();

        Thread threadA = new Thread(volatileClass.new MyThreadA());
        threadA.start();
        Thread.sleep(2000);
        volatileClass.flag = true;
        System.out.println("CurrThread:" + Thread.currentThread().getName() + ",flag:" + volatileClass.flag );
    }

    private class MyThreadA implements Runnable {

        @Override
        public void run() {
            while (!flag) {
                System.out.println("CurrThread:" + Thread.currentThread().getName() + " is Running...");
            }
        }
    }
}

我们根据上面介绍的JMM来分析下,子线程threadA在运行的时候,会把变量flag的值,从主内存拷贝到自己线程的工作内存之中,然后在执行while (!flag)的时候,从工作内存获取值进行判定。在主线程中执行语句volatileClass.flag = true;时,主线程也会先将变量flag的值,从主内存拷贝到自己线程的工作内存当中,然后修改flag=true,在写回主内存当中。所以安装分析,就算在主线程中设置了flag=true,子线程中的循环也不会停止。但是运行的结果,却是停止了,搞不懂哪里出问题了?
那么如果我们想在执行while (!flag)的时候每次都获取主内存中的值呢?这时候就要使用volatile了。更改代码为:

        private boolean flag = false;
替换为:
        private volatile boolean flag = false;

这样,当在主线程中设置flag=true后,循环就会停止了。

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

推荐阅读更多精彩内容