JMM详解,volatile的作用

背景

public class TestThread {
    static int i = 0;

    public static void main(String[] args) throws Exception {
        new Thread(() -> {
            System.out.println("exec");
            while (true) {
                if (i == 2) {
                    break;
                }
            }
            System.out.println("thread done");
        }).start();
        Thread.sleep(1000);
        new Thread(() -> {
            while (true){
                i = 2;
            }
        }).start();

    }
}

这段代码中过,thread done,是一定不会输出的,第一反应应该是原因是cpu缓存,增加volatile就可以输出,原因真的是cpu缓存吗?那么单核CPU会有这个问题吗?实际上单核CPU也会又这个问题,thread done也不会输出

这个问题是由于java内存模型导致的,和CPU内存无关,只不过两者有相似的地方

什么是JMM

image.png

JMM是java内存模型,与JVM没有关系,JMM三大特性,原子性,可见性,有序性,每个线程会把共享变量的副本拷贝到每个线程的本地内存中,所以不一致是JMM上的不一致,在执行i = 2是,CPU其实已经把缓存进行了更新,不过由于Java内存模型的原因更新的只是每个线程的本地变量,并没有更新主内存的变量,同时没有通知其他线程失效缓存

CPU缓存模型及一致性解决方式

image.png

目前基本上服务器为了提高运算能力,都会使用多核CPU,同时由于CPU处理和内存读取的速度相差太大,所以在内存和CPU中间加了3层缓存,这样的话就会存在缓存一致性的问题,比如,cpu1和cpu2同时读取了变量a的值,如果这个时候cpu1把变量a的值修改了,其他cpu下次读取a的时候由于还是从3层缓存中读取,所以还会读取到老的值

为了解决这个问题,通过MESI模型来进行解决,大致的思路是,共享的时候为share状态,当某个cpu要修改的时候改为Exclusive状态即独享状态,同时将通知其他cpu将这个地址的内存信息改为Invalid状态,Invalid状态的时候cpu就会从内存中读取数据,修改后改为Modified状态即修改状态,当某个cpu再次读取的时候,另一个cpu会把Modified状态的内存信息刷到磁盘,同时告诉此cpu的修改后的值(由于cpu之间通信是同步的会严重影响cpu的性能,所以一般会采用MESI的变种模型,增加异步,为了解决异步的问题,这样就有了内存屏障等措施来保证cpu的可以见和重排序的问题)

所以,cpu进行值修改时,对于其他cpu是可见的,上述的问题并不是cpu缓存导致的

JMM

JMM和cpu内存类似,都是有各自的缓存,但是他是CPU上层的模型,即不管是单核还是多核,他就是他,与谁都没有关系,所以导致CPU在修改完成后虽然已经同步给其他cpu同时刷新的内存(线程的本地内存),但是由于JMM的特性导致其他线程读到的还是旧数据,所以为了解决JMM的可见性问题,以及重排序问题,就需要使用volatile或者synchronized来解决,所以也就有了happen-before原则来保证JMM的有序性

分析

JMM中本地内存并不是真实存在的(存在的话回收,内存占用都不好管理)CPU有一致性协议,各个CPU上的缓存都是一致的,为什么上面的代码循环不会退出?
因为即时编译器的原因,可以使用HSDIS查看汇编代码,会发现加volatile和不加volatile的汇编代码不一样

带有volatile的汇编代码:


image.png

不带volatile的汇编代码:


image.png

所以可以看到,带有volatile的每次比较会从内存中取值,不带的会就回死循环

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

推荐阅读更多精彩内容

  • 转自:https://www.cnblogs.com/dolphin0520/p/3920373.htmlvola...
    mayiwoaini阅读 61评论 0 0
  • 本文从计算机模型开始,以及CPU与内存、IO总线之间的交互关系到CPU缓存一致性协议的逻辑进行了阐述,并对JMM的...
    vivo互联网技术阅读 148评论 0 0
  • 现代计算机的内存模型 现代计算机中,cpu的指令速度远超内存的存取速度,由于CPU和内存的运算速度有几个数量级的差...
    传达室马大爷阅读 1,035评论 0 4
  • 我是黑夜里大雨纷飞的人啊 1 “又到一年六月,有人笑有人哭,有人欢乐有人忧愁,有人惊喜有人失落,有的觉得收获满满有...
    陌忘宇阅读 8,523评论 28 53
  • 信任包括信任自己和信任他人 很多时候,很多事情,失败、遗憾、错过,源于不自信,不信任他人 觉得自己做不成,别人做不...
    吴氵晃阅读 6,181评论 4 8