Java 内存模型深度解析

优质博文:IT-BLOG-CN

一、并发编程模型的两个关键问题

【1】并发中常见的两个问题:线程之间如何通信及线程之间如何同步。通信是指线程之间以何种机制来交换信息。在命令式编程中,线程之间的通信机制有两种:内存共享和消息传递
【2】在共享内存的并发模型里,线程之间共享程序的公共状态,通过写-读内存中的公共数据进行隐式通信。在消息传递的并发模型里,如果没有公共状态,线程之间必须通过发送消息来显示进行通讯;
【3】同步是指程序中用于控制不同线程间操作发生相对顺序的机制,可以理解为协同步调,按预定的先后次序运行。这里的“同”字应是指协同、协助、互相配合的意思而非一起的意思。可理解为线程A和B一块配合,A执行到一定程度时要依靠B的某个结果,于是停下来,示意B运行,B执行结束后再将结果给A,A再继续操作。在共享内存并发模型里,同步是显式进行的。在消息传递的并发模型里,由于消息的发送必须在消息接收之前。因此同步是隐式进行的;
【4】Java 的并发采用的是共享内存模型,Java 线程之间的通信总是隐式进行,整个通信过程对程序员完全透明。如果编写多线程程序的 Java程序员不理解隐式进行的线程之间通信的工作机制,很可能会遇到各种奇怪的内存可见性问题;

二、Java内存模型的抽象结构

【1】Java中堆内存主要用于存储实例域、静态域和数组元素等,堆内存在线程之间共享(“共享变量”指实例域、静态域和数组元素)。局部变量(Local Variables),方法定义参数和异常处理器参数(Exception Handler Parameters)不会在线程之间共享,它们不会有内存可见性问题,也不会受内存模型的影响。
【2】Java 线程之间的通讯由 Java内存模型(JMM)控制,JMM决定一个线程对共享变量的写入何时对另一个线程可见。从抽象角度来看,JMM定义了线程和主内存之间的抽象关系:线程之间的共享变量存储在主内存(Main Memory)中,每个线程都有一个私有的本地内存(Local Memory),本地内存中存储了该线程以读/写共享变量的副本。本地内存是 JMM的一个抽象概念,并不真实存在。它涵盖了缓存、写缓存区、寄存器以及其它的硬件和编译器优化。Java内存模型的抽象示意图如下:

从上图来看,如果线程A和线程B之间要通信的话,必须要经历下面2个步骤:
 1)、线程A把本地内存A中更新过的共享变量刷新到主内存中去。
 2)、线程B到内存中去读取线程A之前已更新过的共享变量。
从整体上来看,这两个步骤实质上是线程A在向线程B发送消息,而且这个通信过程必须要经过主内存。JMM通过控制主内存与每个线程的本地内存之间的交互,来为Java程序员提供内存可见性保证。

三、从源代码到指令序列的重排序

在执行程序时,为了提高性能,编译器(javac:将 java源程序编译成中间代码字节码文件)和处理器常常会对指令做重排序。重排序分为三种:
 1)、编译器优化的重排序。编译器再不改变单线程程序语义的前提下,可以重新安排语句的性质顺序。
 2)、指令集并行的重排序。现代处理器采用了指令级并行技术(Instruction-Level Parallelism,ILP)来将多条指令重叠执行。如果不存在数据依赖性,处理器可以改变语句对应机器指令的执行顺序。
 3)、内存系统的重排序。由于处理器使用缓存和读/写缓冲区,这使得加载和存储操作看上去可能是在乱序中执行。从 Java源代码到最终实际执行的指令序列,会分别经历下面重排序:

【1】上述的1属于编译器重排序,2和3属于处理器重排序。这些重排序可能会导致多线程程序出现内存可见性问题。对于编译器,JMM的编译器重排序规则会禁止特定类型的编译器重排序(不是所有的编译器重排序都要禁止)。对于处理器排序,JMM的处理器重排序规则会要求 Java编译器在生成指令序列时,插入特定类型的内存屏障(Memory Barriers,Inter称之为Memory Fence)指令,通过内存屏障指令来禁止特定类型的处理器重排序。
【2】JMM 属于语言级的内存模型,它确保在不同的编译器和不同的处理器平台之上,通过禁止特定类型的编译器重排序和处理器重排序,为程序员提供一致的内存可见性保证。

四、并发编程模型的分类

现在的处理器使用写缓冲区临时保存向内存写入的数据,因为处理器的处理速度远远大于 IO的处理速度。写缓冲区能够保证指令流水线运行,可以有效的避免由于处理器停顿下来等待向内存写入数据而产生的延迟。同时,通过以批处理的方式刷新写缓冲区,以及合并写缓存器中内存地址相同的多次写操作,减少对内存总线的占用。虽然写缓存区有这么多好处,但每个处理器上的写缓冲区,仅仅对它所在的处理器可见,这个特性会对内存操作的执行产生重要的影响:处理器对内存的读/写操作的执行顺序,不一定与内存实际发生的读/写操作顺序一致。举个栗子:

示例列/处理器ProcessAProcessB
代码a=1;//A1x=b;//A2b=2;//B1y=a;//B2
运行结果初始状态a=b=0
处理器运行执行后得到的结果:x=y=0

假设处理器A 和处理器B 按程序的顺序并行执行内存访问,最终可能得到 x=y=0的结果。具体原因:就是上面所说的,处理器上的缓存区仅可见于它所在的处理器。

【1】这里处理器A 和处理器B 可以同时把共享变量写入自己的缓冲区(A1、B1),然后从共享内存中读取另一个共享变量(A2、B2),最后才把自己写缓存中保存的脏数据刷新到内存中(A3、B3),当以这种时序执行时,程序就可以得到 x=y=0的结果。
【2】从内存操作实际发生的顺序来看,直到处理器A执行A3来刷新自己的写缓存区,写操作A1才算真正执行了。虽然处理器A执行内存操作的顺序为:A1——>A2,但实际内存操作的顺序A2——>A1。此时处理器A 的内存操作顺序被重排序了(处理器B相同)
【3】这里关键是,由于写缓存区仅对自己的处理器可见,从而导致处理器执行内存操作的顺序可能会与内存实际的操作执行顺序不一致。由于现在的处理器都支持写缓存,因此现在的处理器都支持写-读操作的重排序。
▶ 常见处理器允许的重排序类型的列表:(N:表示不允许重排序)

处理器/规则Load-LoadLoad-StoreStore-StoreStore(写)-Load(读)数据依赖
SPARC-TSONNNYN
X86NNNYN
IA64YYYYN
PowerPCYYYYN

可以看出常见的处理器都允许 Store-Load重排序,常见的处理器都不允许存在数据依赖的操作重排序。Sparc-TSO 和 X86拥有相对较强的处理器内存模型,它们仅允许对写-读操作做重排序(因为它们都使用了写缓冲区)。为了保证内存可见性,Java编译器在生成指令序列的适当位置会插入内存屏障指令来禁止特定类型的处理器重排序。JMM 把内存屏障指令分为4类:

屏障类型指令示例说明
LoadLoad BarriersLoad1; LoadLoad; Load2确保Load1装载数据优先Load2及所有后续装载指令的装载。
StoreStore BarriersStore1; StoreStore; Store2确保Store1数据对其他处理器可见(刷新到内存)优先Store2及所有后续存储指令的存储。
LoadStore BarriersLoad1; LoadSotre; Store2确保Load1数据装载优先于Store2及所有后续的存储指令刷新到内存。
StoreLoad BarriersStore1; StoreLoad; Load2确保Store1中的数据对其他处理器可见(刷新到内存)优于Load2及所有的后续加载指令。StoreLoad Barriers会使该屏障是前的所有内存访问指令(存储和装载指令)完成后,才执行该屏障后的内存访问指令

StoreLoad Barriers是一个“全能型”的屏障,它同时具有前面3个屏障的效果。现在的大多数处理器都支持该屏障(其他类型的屏障不一定被支持),支持开屏障开销一般很昂贵,因为当前处理器通常要把写缓存中的数据全部刷新到内存中(Buffer Fully Fulsh)。

五、happens-before 简介

JSR-133内存模型使用 happens-before的概念来阐述操作之间的内存可见性。JMM中,如果一个操作执行的结果需要对另一个操作可见,那么这两个操作之间必须要存在 happens-before关系。这两个操作既可以是同一个线程内,也可以是在不同线程之间。
程序员密切相关的 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。

【注意】: 两个操作具有 happens-before关系,并不意味着前一个操作必须要在后一个操作之前执行!happens-before仅仅要求前一个操作(执行的结果)对后一个操作可见,且前一个操作按顺序排在第二个操作之前。happens-before定义很微妙,后文具体说明 happens-before为什么要这么定义。

一个 happens-before规则对应于一个或多个编译器和处理器重排序规则。对于Java程序员来说,happens-before规则简单易懂,它避免了程序员为了理解 JMM提供的内存可见性保证而去学习复杂的重排序规则以及这些规则的具体实现方法。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/636578.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

日常常见应用组件升级记录

一、前言 因近期安全扫描,发现java后端应用涉及多个引用组件版本过低,涉及潜在漏洞利用风险,特记录相关处理升级处理过程,以备后续确认; 二、升级处理过程 2.1、Java类应用内置Spring Boot版本升级 Spring Boot是一…

代码随想录算法训练营第十天|232 用栈实现队列、225 用队列实现栈

栈与队列 栈(stack):先进后出;队列(queue):先进先出栈和队列是STL里面的两个数据结构栈不能遍历元素,提供pop和push等接口,时间复杂度都为O(1)在SGI STL中,如…

python爬虫案例分享

当然,我可以分享一个基本的Python爬虫示例。这个示例将使用Python的requests库来抓取网页内容,然后使用BeautifulSoup库来解析和提取信息。我们将构建一个简单的爬虫来从一个示例网站抓取标题。 Python爬虫示例 目标 提取某网站的标题。 需要的库 r…

bug笔记:解决 HTTP Error 500.30 - ASP.NET Core app failed to start

总结下后端部署windos iis环境net6版本,500.30问题报错的一种解决方案: 一、问题描述 二、解决方案 检查下是否安装了net6对应的环境,是否已经安装 然后在事件管理器>Windows日志>应用程序,里面查看详细异常记录 在iis下面…

使用golang对接微软Azure AI翻译

文章目录 一、官方地址二、准备工作三、代码示例 一、官方地址 https://learn.microsoft.com/zh-CN/azure/ai-services/translator/translator-text-apis?tabsgo 二、准备工作 创建服务 创建服务连接地址:https://portal.azure.com/#create/Microsoft.CognitiveS…

RabbitMQ与SpringAMQP

MQ,中文是消息队列(MessageQueue),字面来看就是存放消息的队列。也就是事件驱动架构中的Broker。(经纪人!) 1.RabbitMQ介绍 微服务间通讯有同步和异步两种方式 同步(通信&#xff0…

ruoyi-cloud—若依微服务打包部署

1. 前端端口修改 2. 后端端口修改 (1)修改ruoyi-gateway服务中的bootstrap.yml的port端口 (2)修改ruoyi-ui中的vue.confing.js的target中的端口 3. 后端部署 (1) 在本地电脑上代码界面上打包后端 在ruoyi项目的bin目录下执行pa…

探索 ChatGPT 中文版:开启自然语言处理新纪元

ChatGPT 中文版是一款由 OpenAI 推出的自然语言处理模型,它在中文语境下展现出了出色的文本生成和对话交互能力。作为程序员,我们对这一领域的创新和发展充满期待。 ChatGPT 中文版不仅能够回答各种技术问题,还能够生成代码示例,…

回溯法:N皇后问题

问题背景 八皇后问题是十九世纪著名的数学家高斯于1850年提出的。 • 问题是:在88的棋盘上摆放八个皇后, 使其不能互相攻击, 即任意两个皇后都不能处于同一行、 同一列或同一斜线上。 • n皇后问题:即在n n的棋盘上摆放n个皇后…

看完这篇我就不信还有人不懂卷积神经网络!

看完这篇我就不信还有人不懂卷积神经网络! 前言 在深度学习大🔥的当下,我知道介绍卷积神经网络的文章已经在全网泛滥,但我还是想要写出一点和别人不一样的东西,尽管要讲的知识翻来覆去还是那么一些,但我想…

Redis原理篇(SkipList)

一.概述 本质是双端链表,只不过在正向遍历时可以不一个一个遍历,而是可以跳着遍历。 怎么实现的呢,下面是SkipList源码 二.源码 1. zskiplist 意义:跳表 zskiplist里面有头指针和尾指针,节点数量,最大…

【信号与系统】(1)连续和离散表示

在信号处理和数学中,连续和离散是两种基本的表示方法,用于描述信号、函数或数据集。 对连续信号 f(t)进行等间隔采样得到 连续表示(Continuous Representation) 连续表示通常用于描述在一个连续范围内变化的信号或函数。在连续…

Java学习(二十一)--JDBC/数据库连接池

为什么需要 传统JDBC数据库连接,使用DriverManager来获取; 每次向数据库建立连接时都要将Connection加载到内存中,再验证IP地址、用户名和密码(0.05s~1s)时间。 需要数据库连接时候,就向数据库要求一个&#xf…

JS-WebAPIS(四)

日期对象(常用) • 实例化 在代码中发现了 new 关键字时,一般将这个操作称为实例化创建一个时间对象并获取时间 获得当前时间 获得指定时间 • 时间对象方法 使用场景:因为日期对象返回的数据我们不能直接使用,所以…

学习心得:二分查找

二分查找 基础:查找元素是否出现 #include <stdio.h> int main() {int a[10]{0,1,1,3,4,5,6,7,8,9},int x;scanf("%d",&x);int l0,r9,count0;while(l<r){int m(lr)/2;if(a[m]x){countm;break;}if(a[m]>x){rm-1;}if(a[m]<x)lm1;}printf("%d…

Elasticsearch 查询语句概述

目录 1. Match Query 2. Term Query 3. Terms Query 4. Range Query 5. Bool Query 6. Wildcard Query 7. Fuzzy Query 8. Prefix Query 9. Aggregation Query Elasticsearch 是一个基于 Lucene 的搜索引擎&#xff0c;提供了丰富的查询DSL&#xff08;Domain Specifi…

【2023我的编程之旅】七次不同的计算机二级考试经历分享

目录 我报考过的科目 第一次报考MS Office 第二次报考Web语言&#xff0c;C语言&#xff0c;C语言 第三次报考C语言&#xff0c;C语言&#xff0c;Java语言 分享一些备考二级的方法 一些需要注意的细节 结语 2023年的CSDN征文活动已经进入了尾声&#xff0c;在这最后我…

Excel·VBA合并工作簿2

其他合并工作簿的方法&#xff0c;见之前的文章《ExcelVBA合并工作簿》 目录 8&#xff0c;合并文件夹下所有工作簿中所有工作表&#xff0c;按表头汇总举例 8&#xff0c;合并文件夹下所有工作簿中所有工作表&#xff0c;按表头汇总 与之前的文章《ExcelVBA合并工作簿&#x…

006.Oracle事务处理

我 的 个 人 主 页&#xff1a;&#x1f449;&#x1f449; 失心疯的个人主页 &#x1f448;&#x1f448; 入 门 教 程 推 荐 &#xff1a;&#x1f449;&#x1f449; Python零基础入门教程合集 &#x1f448;&#x1f448; 虚 拟 环 境 搭 建 &#xff1a;&#x1f449;&…

vue2 点击按钮下载文件保存到本地(后台返回的zip压缩流)

// import ./mock/index.js; // 该项目所有请求使用mockjs模拟 去掉mock页面url下载 console.log(res, res)//token 是使页面不用去登录了if (res.file) {window.location.href Vue.prototype.$config.VUE_APP_BASE_IDSWAPI Vue.prototype.$config.VUE_APP_IDSW /service/mode…