Loading [MathJax]/jax/output/CommonHTML/config.js
前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
社区首页 >专栏 >Java内存模型—JMMhappens-before规则

Java内存模型—JMMhappens-before规则

作者头像
用户1174983
发布于 2018-02-05 07:41:52
发布于 2018-02-05 07:41:52
48400
代码可运行
举报
文章被收录于专栏:钟绍威的专栏钟绍威的专栏
运行总次数:0
代码可运行

 有时候编译器、处理器的优化会导致runtime与我们设想的不一样,为此Java对编译器和处理器做了一些限制,JAVA内存模型(JMM)将这些抽象出来,这样编写代码时就无需考虑那么多底层细节,并保证“只要遵循JMM的规则编写程序,其运行结果一定是正确的”。

JMM的抽象结构

 在Java中,所有的实例、静态变量存储在堆内存中,堆内存是可以在线程间共享的,这部分也称为共享变量。而局部变量、方法定义参数、异常处理参数是在栈中的,栈内存不在线程间共享。

 而由于编译器、处理器的优化,会导致共享变量出现可见性问题,像在多核处理器中(multi-processor),线程可以在不同的处理器上执行,而处理器之间缓存不一致,会使共享变量出现可见性问题,有可能两个线程看到同一个变量不同值。

 JMM将这些硬件做的优化抽象成每个线程都有一个本地内存。需要读写共享变量时,从主内存中拷贝一份到本地内存。当写共享变量时,先写到本地内存中去,在将来某个时间再刷新到主内存中。当再次读共享变量时,则只会从本地内存中读取。

 这样线程间通讯就需要经过两步:

  1. 写线程:刷新本地内存到主内存中去
  2. 读线程:从主内存读取更新后的值

 这样在写-读之间就有一个延迟:本地内存什么时候刷新到主内存中去?导致可见性问题,不同线程可能看到的共享变量不一样。

happens-before

 从字面上看happens-before的意思是“发生在此之前”。这是java对程序执行顺序制定的规则,实现同步必须遵循该规则。这样程序员只需要写出正确的同步程序,happens-before保证运行结果不会错。

A happens-before B,不仅仅表示A在B之前执行,还意味着A的执行结果对B可见,这保证了可见性。

A happens-before B,A也不一定要在B之前执行,如果AB交替,执行结果任然正确,则允许编译器、处理器进行优化重排序。所以只要程序结果正确,编译器、处理器怎么优化,怎么重排序都没问题,都是好的。

happens-before规则

  1. 程序顺序规则:在一个线程中,前面的操作happens-before后面的操作
  2. 锁规则:对同一个锁,解锁happens-before加锁
  3. volatile域规则:写volatile变量,happens-before后面任意一个读这个volatile变量的操作
  4. 传递性:A happens-before B,B happens-before C,则A happens-before C
  5. start()规则:如果线程A执行ThreadB.start() 那么ThreadB.start() happens-before 线程B中任何操作
  6. join()规则:如果线程A执行ThreadB.join(),那么线程B中的所有操作happens-before ThreadB.join()

 下面这个示例有助于理解happens-before

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
double pi = 3.14; //A
double r = 1.0; //B
double area = pi * r *r; //C

 这里有三个happens-before关系,规则1、2是程序顺序规则,规则3是传递性规则推导出来的:

  1. A happens-before B
  2. B happens-before C
  3. A happens-before C

 C依赖于A、B,但是A和B谁也不依赖。所以即使A和B重排序,执行结果也不会发生变化,这种重排序,JMM是运行的。

 下面两种执行顺序的结果都是正确的。

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2017-12-21 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
万丈高楼平地起—Java并发的基石JMM(Java内存模型)
顺序一致性模型可以保证并发编程的特性不被破坏,为多线程程序提供了极强的 内存一致性保证
淇妙小屋
2022/03/30
2490
万丈高楼平地起—Java并发的基石JMM(Java内存模型)
高并发编程-happens-before
happens-before是JMM最核心的概念,理解happens-before是理解JMM的关键
小小工匠
2021/08/17
2400
深入理解Java内存模型(一)——基础
并发编程模型的分类 在并发编程中,我们需要处理两个关键问题:线程之间如何通信及线程之间如何同步(这里的线程是指并发执行的活动实体)。通信是指线程之间以何种机制来交换信息。在命令式编程中,线程之间的通信机制有两种:共享内存和消息传递。 在共享内存的并发模型里,线程之间共享程序的公共状态,线程之间通过写-读内存中的公共状态来隐式进行通信。在消息传递的并发模型里,线程之间没有公共状态,线程之间必须通过明确的发送消息来显式进行通信。 同步是指程序用于控制不同线程之间操作发生相对顺序的机制。在共享内存并发模型里,
小小明童鞋
2018/06/13
6290
java内存模型的理解
可见性定义: 一个线程对共享变量的修改,另外一个线程能够立刻看到,我们称为可见性。
大忽悠爱学习
2023/03/06
3190
java内存模型的理解
Java内存模型
  重排序是指编译器或处理器为了提高程序性能而对指令序列进行重新排序的一种手段。重排序可以导致操作延时或程序看似乱序执行,给程序运行的结果带来一定的不确定性。
在周末
2019/09/11
6520
【死磕Java并发】—–Java内存模型之happens-before
在上篇博客(【死磕Java并发】—–深入分析volatile的实现原理)LZ提到过由于存在线程本地内存和主内存的原因,再加上重排序,会导致多线程环境下存在可见性的问题。那么我们正确使用同步、锁的情况下,线程A修改了变量a何时对线程B可见? 我们无法就所有场景来规定某个线程修改的变量何时对其他线程可见,但是我们可以指定某些规则,这规则就是happens-before,从JDK 5 开始,JMM就使用happens-before的概念来阐述多线程之间的内存可见性。 在JMM中,如果一个操作执行的结果需要对另一个
芋道源码
2018/03/02
6250
【死磕Java并发】—–Java内存模型之happens-before
JMM—详细总结
Java并发的通信机制是通过共享内存实现的。线程之间共享程序的公共状态,线程通过读写内存中的公共状态进行隐式通讯。这对程序员是透明的,我们需要理解其工作机制,以防止内存可见性问题,从而编写出正确同步的代码。
用户5325874
2020/01/16
7150
JMM—详细总结
简述Java内存模型
在介绍Java内存模型(JMM)前,我要打消读者一个错误的认知,那就是JMM与JVM到底是什么关系,现在告诉大家,Java虚拟机模型(JVM)与Java内存模型(JMM)没有本质上的联系。为什么这么说,我来解释一下:想必我的读者大部分都是Java开发工程师,成为一名Java开发工程师必备的两点,就是要了解Java的语法,以及使用Java API,拥有这两点你就可以编写Java代码,编写后的代码需要在Java虚拟机上运行,其实上面我已经把JDK的组成说了出来。JDK(Java Development Kit)就是由Java程序设计语言、Java API类库、Java虚拟机这三部分组成的,是Java程序开发的最小环境(如图2-6所示)。也就是说想要开发Java程序,必备的就是JDK。我们还可以继续把Java API类库分成Java SE API子集和Java虚拟机两部分统称JRE(Java Runtime Environment),JRE是Java程序运行的标准环境。所以说Java虚拟机模型(JVM)是将Java文件编译成class文件并运行class文件的软件,而Java内存模型(JMM)主要定义了线程与内存之间的细节,现在看来两者并没有直接的关系。
胖虎
2020/12/08
4340
Java内存模型相关原则详解
版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
程序新视界
2019/11/12
4020
初步认识线程安全性
这段时间在学习java并发编程这方面的知识,目前正在看《java并发编程之美》,书写的感觉蛮不错的,很细致,个人觉得少了点实际应用场景的案例,但总体来说各方面的知识点写的很详细,从原理到demo都会呈现给读者,此外还看了咕咆的一些教学视屏,里面的讲解让我深有体会,赶紧开个学习专题用以记录自己学习的点点滴滴。
会说话的丶猫
2020/08/11
4090
并发编程原理剖析——深入理解Java内存模型 顶
通信 是指线程之间以何种机制来交换信息。在命令式编程中,线程之间的通信机制有两种:共享内存 和 消息传递。
须臾之余
2019/07/09
5140
Java并发编程(四)Java内存模型
相关文章 Java并发编程(一)线程定义、状态和属性 Java并发编程(二)同步 Java并发编程(三)volatile域 前言 此前我们讲到了线程、同步以及volatile关键字,对于Java的并发编程我们有必要了解下Java的内存模型,因为Java线程之间的通信对于工程师来言是完全透明的,内存可见性问题很容易使工程师们觉得困惑,这篇文章我们来主要的讲下Java内存模型的相关概念。 1.共享内存和消息传递 线程之间的通信机制有两种:共享内存和消息传递;在共享内存的并发模型里,线程之间共享程序的公共状
用户1269200
2018/02/01
7130
Java并发编程(四)Java内存模型
彻底理解Java并发:Java内存模型
进程是静态的概念,进程是资源(CPU、内存等)分配和调度的基本单位,它拥有自己的资源空间,每启动一个进程,系统就会为它分配地址空间;
栗筝i
2022/12/01
3610
彻底理解Java并发:Java内存模型
【Java】【并发编程】详解Java内存模型
Java内存模型即 Java Menory Model,简称JMM。JMM定义了Java虚拟机(JVM)在计算机内存(RAM)中的工作方法。JVM是整个计算机虚拟模型,所以JMM隶属于JVM的。
玖柒的小窝
2021/09/24
2K0
【Java】【并发编程】详解Java内存模型
彻底理解Java并发:volatile关键字
Java 中的 volatile 关键字,用来修饰会被不同线程访问和修改的变量,通常用于并发编程中,是 Java 虚拟机提供的轻量化同步机制。
栗筝i
2022/12/01
4820
彻底理解Java并发:volatile关键字
java内存模型
前言 在学习java多线程并发编程前,必须要了解java内存模型,只有了解java内存模型,才能知道为什么多线程并发时会出现数据不一致,什么时候需要加锁同步等各种问题。下面只是简单阐述下java内存模型及其相关的概念。 内存模型简介 java的并发采用的是共享内存模型(而非消息传递模型)。 Java内存模型(Java Memory Model)描述了Java程序中各种变量(共享变量)的访问规则,以及在JVM中将变量存储到内存和从内存中读取变量这样的底层细节。 Java线程之间的通信由Java内存模型(JMM
java404
2018/05/18
1K0
​带你深入理解Java内存模型JMM
在并发编程中,我们需要处理两个关键问题:线程之间如何通信及线程之间如何同步(这里的线程是指并发执行的活动实体)。通信是指线程之间以何种机制来交换信息。在命令式编程中,线程之间的通信机制有两种:共享内存和消息传递。
Java技术江湖
2019/09/25
4230
深入理解并发编程艺术-内存模型篇
随着硬件技术的飞速发展,多核处理器已经成为计算设备的标配,这使得开发人员需要掌握并发编程的知识和技巧,以充分发挥多核处理器的潜力。然而并发编程并非易事,它涉及到许多复杂的概念和原理。为了更好地理解并发编程的内在机制,需要深入研究内存模型及其在并发编程中的应用。本文将主要以Java内存模型来探讨并发编程中BUG的源头和处理这些问题的底层实现原理,助你更好地把握并发编程的内在机制。
卡卡罗特杨
2023/06/26
5920
全面理解Java内存模型
Java内存模型即Java Memory Model,简称JMM。JMM定义了Java 虚拟机(JVM)在计算机内存(RAM)中的工作方式。JVM是整个计算机虚拟模型,所以JMM是隶属于JVM的。
全栈程序员站长
2022/07/22
4080
全面理解Java内存模型
😀 Java并发 - (并发基础)
用户7630333
2023/12/07
2000
😀 Java并发 - (并发基础)
相关推荐
万丈高楼平地起—Java并发的基石JMM(Java内存模型)
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
查看详情【社区公告】 技术创作特训营有奖征文
本文部分代码块支持一键运行,欢迎体验
本文部分代码块支持一键运行,欢迎体验