Java内存模型

什么是可见性
可见性:一个线程对共享变量值的修改,能够及时地被其他线程看到。

共享变量:如果一个变量在多个线程的工作内存中都存在副本,那么这个变量就是这几个线程的共享变量

并发编程的两个关键问题

  1. 线程之间如何通信(以何种机制来交换信息)
  2. 线程之间如何同步

线程之间的通信机制有两种

  1. 共享内存,线程共享程序的公共状态,通过写-读公共状态达到隐式通信 。
  2. 消息传递,线程之间没有公共状态,线程之间必须通过发送消息来显示进行通信。java中典型的消息传递方式就是wait()和notify()。

同步
指的是程序中用于控制不同线程间操作发生相对顺序的机制。

  1. 在共享内存并发模型中,同步是显示进行的,因为程序员必须显示的指定某个方法或者某段代码在线程之间是互斥执行的。
  2. 在消息传递并发模型中,同步是隐式进行的,因为消息发送必须在消息接受之前。

JAVA的并发采用的是共享内存模型,线程之间的通信总是隐式的,通信过程对于程序员完全透明。

内存模型的抽象结构

java中素所有实例域、静态域和数组元素都存储在堆内存中,堆内存是线程共享的。局部变量、方法定义参数、异常处理器参数不会再线程之间共享,不会有内存可见性问题。

Java线程间通信由Java内存模型(JMM)控制,JMM决定了一个线程对共享变量的写入何时对另一个线程可见。

JMM定义了线程和主内存之间的抽象关系:共享变量存储在主内存中,每个线程都有一个本地内存,本地内存中存储了共享变量的副本,本地内存是JMM的一个抽象概念并不真实存在。 Java的并发采用的是共享内存模型

线程AB之间通信 就是 线程A把本地内存A中的更新后的共享变量刷新到主内存中 然后线程B到主内存中去读取线程A之前已更新过的共享变量,实质上是线程A向线程B发送消息。

JMM通过控制主内存和每个线程的本地内存之间的交互,来为java程序员提供内存可见性保证。

重排序

重排序是指编译器和处理器为了优化程序性能而对指令序列进行重新排序的一种手段。
重排序分为三种类型

  1. 编译器优化的重排序,编译器在不改变单线程程序语义的前提下,可以重新安排语句的执行顺序。
  2. 指令级并行的重排序。 将多条指令重叠执行,如果不存在数据依赖性,处理器可改变语句对应机器指令的执行顺序。
  3. 内存系统重排序。由于处理器使用缓存和读/写缓冲区,使得加载和存储操作看上去可能是乱序执行的。

1属于编译器重排序 2,3属于处理器重排序

内存屏障指令
为了保证内存可见性,java编译器在生成指令序列的适当位置会插入内存屏障指令来禁止特定类型的处理器重排序。内存屏障有四种
load 简称l store简称 Barriers简称b

  1. llb
  2. ssb
  3. lsb
  4. slb

排列顺序与执行的先后顺序相同
其中slb是比较全能的一个屏障 它确保store1数据对其他处理器变得可见(指刷新到内存)先于load2
及所有后续装载指令的装载 并且会使这个屏障之前所有内存访问之灵完成之后在赤星该屏障之后的内存访问指令
同时具备了其他三个屏障的效果,但是这个屏障开销昂贵,因为处理器会把写缓存中的所有数据全部刷新到内存中去。

数据依赖性
编译器和处理器不会改变存在数据依赖关系的两个操作的执行顺序
什么是数据依赖性?
如果两个操作访问同一个变量且这两个操作中有一个为写操作,此时两个操作之间就存在数据依赖性

happens-before

JSR-133内存模型使用这个概念来阐述操作之间的内存可见性,在JMM中如果一个操作的执行结果需要对另一个操作看见,那么这两个操作之间必须存在 happens-before关系
happens-before有两个规则

  1. 程序顺序规则,一个线程中的每个操作happens-before后续操作
  2. 监视器锁规则,对一个锁的解锁happens-before这个锁的加锁
  3. volatile变量规则,对一个volatile域的写 happens-before对这个变量的读
  4. 传递性 a>b b>c a>c。。
    happens-before关系 并不是说 A h-b B A就一定先于B执行 而是要求A的执行结果对B可见。

只要多线程的程序是正确同步的,JMM保证该程序在任意处理器平台上的执行结果与改程序在顺序一致性内存模型中的执行结果一致。

Volatile

是java虚拟机提供的最轻量级的同步机制。它有两层语义

  1. 保证次变量对所有线程的可见性,这里可见性是指当一条线程修改了这个变量的值,新值对于其它线程来说就是立即可得知的。

    由于Volatile变量只能保证可见性 以此以下两种场景,仍然需要加锁来保证原子性

    1. 运算结果依赖变量的当前值,如(i++),或者不能确保只有单一的线程修改变量的值
    2. 变量需要与其他的状态变量共同参与不变约束。
  2. 禁止指令重排序优化

Java内存模型特征

1. 原子性
由java内存模型来直接保证的原子性变量操作包括read,load,assign,use,store,write.如果场景需要更大范围的原子性保证,java内存模型还提供了lock和unlock操作来满足需求,但是并没有把这两个操作直接开放给用户使用,而是提供了更高层次的字节码指令monitorenter,monitorexit来隐式地使用这两个操作,这两个字节码指令反映到代码中就是同步快 synchronized关键字,在这个块里面的操作具有原子性
2.可见性
一个线程对共享变量值的修改,其它线程能够立即得知这个修改,java内存模型是通过在变量修改后将新值同步回主内存,在变量读取前从主内存刷新变量值这种依赖主内存作为传递媒介的方式来实现可见性的。

volatile变量和普通变量的区别是volatile变量的特性保证了新值能够立即同步到主内存,每次使用立即从主内存刷新。因此可以说volatile保证了多线程操作时变量的可见性,而普通变量不能保证这一点。

除了volatile,synchronized和final也能实现可见性。

同步块之所以获取可见性是因为对一个变量执行unlock操作之前,必须先把此变量同步到主内存中

对于final变量,因为final字段在构造器中一旦初始化完成,并且构造器没有吧this的引用传递出去,那在其他线程中就能够看到final字段的值

3. 有序性
如果在本线程中观察,所有的操作都是有序的;如果在一个线程中观察另一个线程,所有的操作都是无序的,前半句是 指线程内表现为串行 的语义,后半句是指指令重排序和工作内存和主内存同步延迟现象。

java提供volatile和synchronized两个关键字来保证线程之间操作的有序性。volatile本身就禁止指令重排序,synchronized又规定一个变量在同一时刻只允许一条线程对其进行lock操作,这就决定了持有同一个锁的两个同步块只能串行地进入

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

推荐阅读更多精彩内容