volatile简单介绍

volatile

作用

  • 内存可见性

    public class VolatileTest {
        //(1)定义一个变量a
        int a = 0;
        void test() {
            System.out.println("a=" + a + " start");
            //(2)如何a=0就一直循环
            while (a == 0) {
            }
            System.out.println("a=" + a + " end");
        }
    
        public static void main(String[] args) {
            VolatileTest t = new VolatileTest();
            //(3)创建一个线程执行test方法
            Thread thread1 = new Thread(() ->{
                t.test();
            });
            thread1.start();
          
            try {
                TimeUnit.SECONDS.sleep(1);
            } catch (Exception ex) {
                ex.printStackTrace();
            }
          
            //(4)休眠一秒后修改a的值为1
            t.a = 1;
              //(5)输出a的值
              System.out.println("a=" + t.a);
        }
    
    }
    

    此程序

    (1)中定义了一个变量a,赋值为0;

    (2)中定义了一个test方法,方法体内有一个while循环,判断a的值是否为0,如果为0,就一直空循环;

    (3)的main方法创建了一个线程,线程去执行test方法;

    (4)在主线程休眠1秒后将a的值改为1;

    (5)输出a的最新值

    当控制台输出了a=1时,表示a的值已经被修改为了1,所以正常来说程序应该输出“a=1 end”结束,但是实际情况并没有。

  • 原因:
    1、在main方法中新开启的线程thread1,thread1读取主内存中的a=0到自己的工作内存中;
    2、然后main线程从主内存中获取a的值,判断a的值是是否为0,因为主内存中的a=0没有别的程序变动过,所以a肯定为0;
    3、然后main线程将a的值改为1,并刷新到主内存中,这会输出a=1;
    4、但是在thread1线程中,它是不知道主内存中的a的值被修改为1了,目前它还在使用自己工作内存中的a的值,也不知道什么时候thread1线程会去主内存中重新获取a的值,所以这里不会输出“a=1 end”结束。
    因为这里存在内存不可见问题:即main线程修改了主内存a的值,但是thread1线程无法感知到a的值被修改了。想一下,其实解决这个问题很简单,无非就是在main线程修改了a的值,并且会写到主内存之后,thread1可以知道自己工作内存中的这个a的值现在是无效的了,如果需要使用这个a,必须去主内存中重新读取a的值。刚好在Java中volatile关键字就有这种效果。volatile主要是使用计算机的M(被修改Modified)E(独享的Exclusive)S(共享的Shared)I(无效的Invalid)缓存一致性来处理这种问题。
    1、在thread1读取到a的之后,在thread1的工作内存中a会被标志为E(独享)
    2、在main线程读取了a之后,main的工作内存以及thread1的工作内存的a都会被标志为S(共享)
    3、在main线程修改了a的值,main内存中的a会被标志为M(被修改),并写入主内存之后,thread1的工作内存中的a会被标志为I(无效)
    4、在thread1的循环中,在读取a的时候,发现自己的工作内存中的a是无效的,就会去主内存中重新读取a的值,这个时候工作内存中的a又会被标志为S(共享)

    优化代码:在变量a的前面加上volatile;

    volatile int a = 0;
    
  • 防止指令重排序

    public class SingletonTest {
    
        private String word = "";
    
        public static volatile SingletonTest INSTANCE = null;
    
        private SingletonTest() {
            word = "Hello World";
            System.out.println(word);
        }
    
        /**
         * 双重检查获取单例(DCL)
         * @return new SingletonTest()
         */
        public static SingletonTest getInstance() {
            //(1)第一重检查
            if (INSTANCE == null) {
                //(2)加锁
                synchronized (SingletonTest.class) {
                    //(3)第二重检查
                    if (INSTANCE == null) {
                          //(4)获取对象
                        INSTANCE = new SingletonTest();
                    }
                }
            }
            return INSTANCE;
        }
    
        public static void main(String[] args) {
            SingletonTest singletonTest = getInstance();
        }
    }
    

    以上是一个DCL(双重检查)方式获取单例的代码。但是为什么还需要在这里加上volatile

    public static volatile SingletonTest INSTANCE = null;
    

    原因是在(4)中

    INSTANCE = new SingletonTest();
    

    这段代码在计算机底层是分了三步

    1、给SingletonTest开辟一块内存;
    2、给SingletonTest赋初始值;
    3、new SingletonTest() 赋值给INSTANCE;
    这三步中很渺小可能执行顺序为1>3>2;当3比2先执行时,这个时候当别的线程执行到了(1)时,其实INSTANCE已经不是null了,然后就返回了。很明显这个是不对的。

    • volatile如何保证有序性:

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

    2)在进行指令优化时,不能将在对volatile变量访问的语句放在其后面执行,也不能把volatile变量后面的语句放到其前面执行。

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

推荐阅读更多精彩内容