查看“Java 重排序对多线程的影响”的源代码
←
Java 重排序对多线程的影响
跳到导航
跳到搜索
因为以下原因,您没有权限编辑本页:
您所请求的操作仅限于该用户组的用户使用:
用户
您可以查看和复制此页面的源代码。
<syntaxhighlight lang="java"> public class ReorderExample { int a = 0; boolean flag = true; public void writer() { a = 1; // 1 flag = true; // 2 } public void reader() { if (flag) { // 3 int i = a * a; // 4 } } } </syntaxhighlight>flag 变量是个标记,用来标识变量 a 是否已被写入。 这里假设有两个线程 A 和 B,A 首先执行 writer() 方法,随后 B 线程接着执行 reader() 方法。 线程 B 在执行操作 4(<code>int i = a * a;</code>)时,能否看到线程 A 在操作 1(<code>a = 1;</code>) 对共享变量 a 的写入呢? '''<big>答案是:不一定能看到</big>'''。 === 操作 1 和 操作 2 重排序 === 由于操作 1 和操作 2 没有数据依赖关系,编译器和处理器可以对这两个操作重排序; 同样,操作 3 和操作 4 没有数据依赖关系,编译器和处理器也可以对这两个操作重排序。 当操作 1 和操作 2 重排序时,可能会产生什么效果?看下图: [[文件:重排序程序执行时序图3.png|无|缩略图|494x494像素|虚箭线标识错误的读操作]] 如图所示,操作 1 和 操作 2 做了重排序。 程序执行时,线程 A 首先写标记变量 flag,随后线程 B 读这个变量。 由于条件判断为真,线程 B 将读取变量 a。 此时,变量 a 还没有被线程 A 写入,在这里多线程程序的语义就被重排序破坏了! === 操作 3 和操作 4 重排序 === 下面看看当操作 3 和 操作 4 重排序时会产生什么效果(借助这个重排序,可以顺便说明控制依赖性)。 [[文件:重排序程序执行4.png|无|缩略图|662x662像素]] 在程序中,操作 3和操作 4 存在控制依赖关系。 当代码中存在控制依赖时,会影响指令序列执行的并行度。 为此,编译器和处理器会采用猜测(Speculation)执行来克服控制相关性并对并行度的影响。 以处理器的猜测执行为例,执行线程 B 的处理器可以提前读取并计算 a*a,然后把计算结果临时保存到一个名为重排序缓冲(Reorder Buffer,ROB)的硬件缓冲中。 当操作 3 的条件判断为真时,就把该计算结果写入变量 i 中。 从上图可以看出,猜测执行实质上对操作 3和 4 做了重排序。 重排序也在这里破坏了多线程程序的语义! 在单线程程序中,对存在控制依赖的操作重排序,不会改变执行结果(这也是 as-if-serial 语义允许对存在控制依赖的操作做重排序的原因); 但在多线程程序中,对存在控制依赖的操作重排序,可能会改变程序的执行结果。
返回至
Java 重排序对多线程的影响
。
导航菜单
个人工具
登录
名字空间
页面
讨论
变种
视图
阅读
查看源代码
查看历史
更多
搜索
导航
首页
Spring Boot 2 零基础入门
Spring Cloud
Spring Boot
设计模式之禅
VUE
Vuex
Maven
算法
技能树
Wireshark
IntelliJ IDEA
ElasticSearch
VirtualBox
软考
正则表达式
程序员精讲
软件设计师精讲
初级程序员 历年真题
C
SQL
Java
FFmpeg
Redis
Kafka
MySQL
Spring
Docker
JMeter
Apache
Linux
Windows
Git
ZooKeeper
设计模式
Python
MyBatis
软件
数学
PHP
IntelliJ IDEA
CS基础知识
网络
项目
未分类
MediaWiki
镜像
问题
健身
国债
英语
烹饪
常见术语
MediaWiki帮助
工具
链入页面
相关更改
特殊页面
页面信息