Android---java内存模型与线程

Java 内存模型翻译自 Java Memory Model,简称 JMM。它所描述的是多线程并发、CPU 缓存等方面的内容。

在每一个线程中,都会有一块内部的工作内存,这块内存保存了主内存共享数据的拷贝副本。但在 Java 线程中并不存在所谓的工作内存(working memory),它只是对 CPU 寄存器和高速缓存的抽象描述

CPU 普及

线程是 CPU 调度的最小单位,线程中的字节码指令最终都在 CPU 中执行。CPU 在执行的时候,免不了要和各种数据打交道,而 Java 中所有数据都是放在主内存中的。随着 CPU 的发展,CPU 的执行速度越来越快,但内存的技术并没有太大变化,所以在内存中读取和写入数据的过程和 CPU 执行速度比起来,差距会越来越大。如上图箭头所示,CPU 对主内存的访问需要等待较长的时间,这样就体现不出 CPU 超强的运算能力了。

因此为了压榨处理器性能,达到高并发的效果,在 CPU 中添加了高速缓存 cache 来作为缓冲。在执行运算时,CPU 会将运算所使用到的数据复制到高速缓存中,让运算能够快速进行。当运算结束后,再将缓存中的结果刷回主内存。这样 CPU 就不用等待主内存的读写操作了。如下图所示:

 上面的方案,看起来一切正常。但问题也随之而来,每个 CPU 处理器都有自己的高速缓存,同时又共同操作同一块主内存,当多个处理器同时操作主内存时,可能导致数据不一致,这就是缓存一致性问题

 缓存一致性问题

现在,市面上的手机通常有2个或多个 CPU,其中一些 CPU 还有多核。每个 CPU 在某一时刻都能运行一个线程。如果 Java 程序是多线程的,就有可能存在多个线程在同一时刻被不同的 CPU 执行的情况

如下代码所示:

public int x = 0;
public int y = 0;Thread p1 = new Thread(){public void run(){int r1 = x;y = 1;}
};Thread p2 = new Thread(){public void run(){int r2 = y;x = 2;}
};p1.start();
p2.start();

假设我们的一台设备上有2个CPU,分别为 C1 和 C2,将上面这段代码执行在这台设备上,最后打印出的 r1 和 r2 值分别是多少?答案是不确定的。

情况1:r1 = 0, r2 = 1

假设 P1 现在 C1 中执行完毕,并成功刷新回主内存中,此时 r1 = 0, x = 0, y =1。然后 P2 在 C2 中执行,从主内存中加载 y =1 ,并赋值给 r2。此时 r2 = 1, x = 2, y = 1, r1 = 0。

情况2:r1 = 2, r2 = 0

假设 P2 现在 C1 中执行完毕,并成功刷新回主内存中,此时 r2 = 0, x = 2, y =0。然后 P1 在 C2 中执行,从主内存中加载 x =2 ,并赋值给 r2。此时 r1= 2, x = 2, y = 1, r2 = 0。

情况3:r1 = 0, r2 = 0(特殊)

x, y的值分别缓存在 C1 和 C2 的缓存中,首先 P1 在 C1 中执行完毕,但是并未将结果刷新回主内存中,此时主内存中的 x = 0, y = 0。然后 P2 在 C2 中执行,缓存中的 y = 0, 将其赋值给 r2,此时,r2 = 0 , x = 2, y = 1,如下图所示:

虽然 C1 和 C2 的主内存中修改了 x, y 的值,但并未将它们刷新到主内存中。这就是缓存一致性问题

指令重排

除了缓存一致性问题,还存在另外一种硬件问题,即指令重排。为了使 CPU 内部的运算单元能够尽量充分利用,处理器可能会对输入的字节码指令进行重排序处理,也就是处理器优化。比如 Java 虚拟机的即时编译器(JIT),如以下代码,

a = 1;
b = 2; 
a = a + 1

编译后的字节码指令如下:

可以看出,在上述红框中的指令是表达的同一种语义,并且指令7并不依赖指令2和3,在这种情况下,CPU 会对指令的顺序做如下优化。

从 Java 语言的角度看这种优化如下:

也就是说在 CPU 层面,Java 有时候并不会严格按照文件中的顺序去执行。

缓存一致性问题和指令重排的内容表明,如果我们任由 CPU 优化,那么我们编写的 java 代码所执行的效果会大大出乎我们的意料。

为了解决这个问题,让 Java 代码在不同硬件、不同操作系统中输出的结果达到一致。Java 虚拟机规范提出了一套机制--> java 内存模型

Java 内存模型

内存模型是一套共享内存系统中多线程读写操作行为的规范。这套规范屏蔽了底层各种硬件和操作系统的内存访问差异。解决了 CPU 多级缓存、CPU优化、指令重排等导致的内存访问问题,从而保证 Java 程序(尤其是多线程程序)在各种平台下对内存的访问效果一致。

在 java 内存模型中,统一用工作内存来当作 CPU 中寄存器或高速缓存的抽象。线程之间的共享变量存储在主内存中,每个线程都有一个私有工作内存(类比 CPU 中的寄存器或者高速缓存),本地工作内存中存储了该线程读/写共享变量的副本。

在这套规范中有一个非常重要的规则 happens-before 先行发生原则。

happens-before 先行发生原则

happens-before 用于描述两个操作的内存可见性,通过保证可见性的机制可以让应用程序免于数据竞争干扰。

定义如果一个操作 A happens-before 另一个操作 B,那么操作 A 的执行结果将对操作 B 可见

反过来理解,如果操作 A 的结果必须对操作 B 可见,那么操作 A 必须 happens-before 操作 B。

举例:

private int value = 0;public void setValue(int value){value = 1;
}public int getValue(){return value;
}

假设 setValue 就是操作 A,getValue 就是操作 B,如果先后在两个线程中调用 A 和 B,那最后在 B 操作中返回的 value 值是多少呢?

情况1:A happens-before B 不成立

当线程调用操作 B 时,即使操作 A 已经在其它线程中被调用过,并且 value 也被成功设置为1。但这个修改对操作 B 仍然是不可见的,根据上面 CPU 缓存的内容, value 值有可能返回1,也有可能返回0。

情况2:A happens-before B 成立

根据 happens-before 的定义,先行发生动作的结果对后续动作是可见的,也就是先执行 A  后的结果对后续的操作 B 是始终可见的。即,先调用 setValue() 将 value 的值修改为1,后续在其它线程中调用 getValue() 获得的 value 值一定是1。

在 Java 中的两个操作如何算符合 happens-before 规则了呢?

JMM 中定义了以下几种情况是自动符合 happens-before 规则的。

1. 程序次序规则

在单线程内部,如果一段代码的字节码顺序也隐私符合 happens-before 原则,那么逻辑顺序靠前的字节码执行结果一定是对后续逻辑字节码可见。如下代码所示:
 

int a = 10; // 1
b = b + 1; //2

当代码执行到2处时,a = 10 这个结果已经是公之于众的,至于用没用到 a  这个结果则不一定,比如上面的代码中没有用到 a = 10 这个结果。

2. 锁定规则

一个锁如果处于被锁定状态,那么必须先执行 unlock 操作后才能进行 lock 操作。

3. 变量规则

volatile 保证了线程可见性。如果一个线程先写了一个 volatile 变量,然后另一个线程去读这个变量,那么这个写操作一定是 happens-before 读操作的。

4. 线程启动规则

Thread 对象的 start() 方法先行发生于此线程的每一个动作。假定线程 A 在执行过程中,通过执行 ThreadB.start() 来启动线程 B,那么线程 A 对共享变量的修改在线程 B 开始执行后确保对线程 B 可见。

4. 线程中断规则

对线程interrupt()方法的调用先行发生于被中断线程的代码检测,直到中断事件的发生。

5. 线程终结规则

线程中所有的操作都发生在线程的终止检测之前,可以通过Thread.join()方法结束、Thread.isAlive() 的返回值等方法检测线程是否终止执行。假定线程A在执行的过程中,通过调用ThreadB.join()等待线程B终止,那么线程B在终止之前对共享变量的修改在线程A等待返回后可见。

6. 对象终结规则

一个对象的初始化完成发生在它的finalize()方法开始前。

happens-before 原则具有传递性

如果操作 A happens-before 操作 B,而操作 B happens-before 操作 C,那么操作 A 一定 happens-before 操作 C。

Java 内存模型应用

根据 happens-before 原则,能够解决在并发环境下操作之间是否可能存在冲突的所有问题。可以可以通过 Java 提供的一系列关键字,将实现的多线程操作“happens-before 化”。happens-before 化”是将本来不符合 happens-before 原则的某些操作,通过某种手段使它们符合 happens-before 原则。

方法1:使用 volatile 关键字

private volatile int value = 0;public void setValue(int value){value = 1;
}public int getValue(){return value;
}

方法2:使用 synchronized 关键字修饰操作

private int value = 0;public void setValue(int value){synchronized{value = 1;}
}public int getValue(){synchronized{return value;}
}

通过以上两个方式都可以使 setVaule() 和 getValue() 符合 happens-before 原则。当在某一线程中调用 setVaule()后,再在其它线程中调用 getValue() 获取到的值一定是 1。

总结

Java 内存模型的来源:

主要是因为 CPU 缓存指令重排等优化操作造成多线程程序结果不可控。

Java 内存模型是什么:

本质上它就是一套规范,在这套规范中有一条最重要的 happens-before 原则。

Java 内存模型的使用

简单介绍了两种方式:volatile 和 synchronized。
 

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

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

相关文章

C#实现OPC DA转OPC UA服务器

运行软件前提前安装好OPC运行组件: 为方便演示,提前准备好了一个DAServer服务器: 接下来开始配置: 该软件主要实现的功能如下: 配置过程也相对简单: 第一步: 编辑如下文件: 第二步…

YOLOv7改进策略:RIFormerBlock助力检测|CVPR2023 RIFormer:无需TokenMixer也能达成SOTA性能的极简ViT架构

💡💡💡本文属于原创独家改进: 稀疏重参数RIFormerBlock模型引入YOLOv7进行创新性 RIFormerBlock | 亲测在多个数据集实现涨点; 收录: YOLOv7高阶自研专栏介绍: http://t.csdnimg.cn/tYI0c ✨✨✨前沿最新计算机顶会复现 🚀🚀🚀YOLOv7自研创新结合,轻松…

redis在linux系统的安装与使用

一、单机安装Redis。 1.安装redis依赖 在控制台输入 yum install -y gcc tcl2.上传安装包 下载好的安装包上传到/usr/local/src/ 上传方法: 1.确保你拥有Linux服务器的IP地址、用户名和密码。 2.在Windows上,打开命令提示符(Command Promp…

大数据学习(7)-hive文件格式总结

&&大数据学习&& 🔥系列专栏: 👑哲学语录: 承认自己的无知,乃是开启智慧的大门 💖如果觉得博主的文章还不错的话,请点赞👍收藏⭐️留言📝支持一下博>主哦&#x…

Flink---14、Flink SQL(SQL-Client准备、流处理中的表、时间属性、DDL)

星光下的赶路人star的个人主页 你生而真实,而非完美 文章目录 1、Flink SQL1.1 SQL-Client准备1.1.1 基于yarn-session模式1.1.2 常用配置 1.2 流处理中的表1.2.1 动态表和持续查询1.2.2 将流转换为动态表1.2.3 用SQL持续查询1.2.4 将动态表转换为流 1.3 时间属性1.…

PLC和工控机的网络特性

现场总线技术是工业自动化***深刻变革之一。PLC和工控机采用现场总线后可方便地作为I/O站和监控站连接在DCS系统中。现场总线是一种取代4~20mA标准,用于连接智能现场设备和控制设备的双向数字通讯技术,现场总线具有开放性和互操作性&#xff…

【重拾C语言】八、表单数据组织——结构体(类型、类型别名、直接/间接访问;典例:复数、成绩单)

目录 前言 八、结构体 8.1 结构体类型 8.2 结构体类型名 8.2.1 typedef关键字 8.2.1 结构体类型别名 8.3 结构体变量 8.3.1 使用结构体类型引用 8.3.2 使用结构体类型定义 8.3.3 使用typedef定义的结构体类型别名 8.4 访问结构体变量 8.4.1 直接成员选择表达式 8.…

leetcode 打家劫舍篇

198. 打家劫舍 你是一个专业的小偷,计划偷窃沿街的房屋。每间房内都藏有一定的现金,影响你偷窃的唯一制约因素就是相邻的房屋装有相互连通的防盗系统,如果两间相邻的房屋在同一晚上被小偷闯入,系统会自动报警。 给定一个代表每个…

yolov8 strongSORT多目标跟踪工具箱BOXMOT

1 引言 多目标跟踪MOT项目在Github中比较完整有:BOXMOT , 由mikel brostrom提供。在以前的版本中,有yolov5deepsort(版本v3-v5), yolov8strongsort(版本v6-v9),直至演变…

[Machine learning][Part4] 多维矩阵下的梯度下降线性预测模型的实现

目录 模型初始化信息: 模型实现: 多变量损失函数: 多变量梯度下降实现: 多变量梯度实现: 多变量梯度下降实现: 之前部分实现的梯度下降线性预测模型中的training example只有一个特征属性&#xff1a…

【k8s】ingress-nginx通过header路由到不同后端

K8S中ingress-nginx通过header路由到不同后端 背景 公司使用ingress-nginx作为网关的项目,需要在相同域名、uri,根据header将请求转发到不同的后端中在稳定发布的情况下,ingress-nginx是没有语法直接支持根据header做转发的。但是这个可以利…

Tmux教学【有图有代码】

Tmux教学【有图有代码】 0、前言1、Tmux基本概念1.1 安装 2、Tmux使用2.1 session操作2.2 window操作2.3 pane操作2.4 其他操作 3、日常中Tmux的工作流 0、前言 想必同学们在跑代码时也许会存在这样一个问题: 本地在连接远程服务器跑代码的时候,本地ssh窗…

分类预测 | MATLAB实现KOA-CNN-GRU开普勒算法优化卷积门控循环单元数据分类预测

分类预测 | MATLAB实现KOA-CNN-GRU开普勒算法优化卷积门控循环单元数据分类预测 目录 分类预测 | MATLAB实现KOA-CNN-GRU开普勒算法优化卷积门控循环单元数据分类预测分类效果基本描述程序设计参考资料 分类效果 基本描述 1.MATLAB实现KOA-CNN-GRU开普勒算法优化卷积门控循环单…

Unity编辑器从PC平台切换到Android平台下 Addressable 加载模型出现粉红色,类似于材质丢失的问题

Unity编辑器在PC平台下使用Addressable加载打包好的Cube,运行发现能正常显示。 而在切换到Android平台下,使用Addressable时加载AB包,生成Cube对象时,Cube模型呈现粉红色,出现类似材质丢失的问题。如下图所示。 这是…

【机器学习】决策树原理及scikit-learn使用

文章目录 决策树详解ID3 算法C4.5算法CART 算法 scikit-learn使用分类树剪枝参数重要属性和接口 回归树重要参数,属性及接口交叉验证代码示例 一维回归的图像绘制 决策树详解 决策树(Decision Tree)是一种非参数的有监督学习方法,…

【算法优选】 二分查找专题——贰

文章目录 😎前言🌲[山脉数组的峰顶索引](https://leetcode.cn/problems/peak-index-in-a-mountain-array/)🚩题目描述:🚩算法思路🚩代码实现: 🌴[寻找峰值](https://leetcode.cn/pro…

VUE3基础知识梳理

VUE3基础知识梳理 一、vue了解和环境搭建1.vue是什么:cn.vuejs.org/vuejs.org2.渐进式框架3.vue的版本4.vueAPI的风格5.准备环境5.1.创建vue项目5.2.vue的目录结构 二、vue3语法1.干净的vue项目2.模板语法2.1 文本插值2.2属性绑定2.3条件渲染2.4列表渲染2.5通过key管…

10、SpringBoot_测试用例

四、测试用例 1.准备工作 添加依赖 <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-devtools</artifactId><optional>true</optional></dependency><dependency><groupId>com…

ELK 处理 SpringCloud 日志

在排查线上异常的过程中&#xff0c;查询日志总是必不可缺的一部分。现今大多采用的微服务架构&#xff0c;日志被分散在不同的机器上&#xff0c;使得日志的查询变得异常困难。工欲善其事&#xff0c;必先利其器。如果此时有一个统一的实时日志分析平台&#xff0c;那可谓是雪…

2023年电工(初级)证考试题库及电工(初级)试题解析

题库来源&#xff1a;安全生产模拟考试一点通公众号小程序 2023年电工&#xff08;初级&#xff09;证考试题库及电工&#xff08;初级&#xff09;试题解析是安全生产模拟考试一点通结合&#xff08;安监局&#xff09;特种作业人员操作证考试大纲和&#xff08;质检局&#…