内存屏障1

news2025/1/12 0:53:31

内存屏障

引入

我们知道 volatile 能保证 JMM约束的 可见性和有序性。

关于有序性,到底该如何理解?

有序性的根本保证,就是 禁止指令重排序

重排序

重排序是指 编译器和处理器 为了优化程序性能 而对指令序列进行重新排序的一种手段,有时候会改变程序语句的先后顺序。

  • 不存在数据依赖关系,可以重排序
  • 存在数据依赖关系,禁止重排序
    但是重排序后的指令,绝不能改变原有的串行语义,这点在并发编程中 重点考虑

概念

内存屏障
也称为内存栅栏,屏障指令等,是一类同步屏障指令 ,是CPU或编译器在对内存随机访问的操作中的一个同步点,使得此点之前的所有读写操作都执行后,才可以执行此点之后的操作。避免了代码重排序。

内存屏障其实是一种JVM指令,java内存模型的重排规则会要求Java编译器在生成JVM指令时 插入特定的内存屏障指令,通过这些内存屏障指令,volatile实现了java内存模型中的可见性和有序性(禁止重排序),但Volatile无法保证原子性。

  • 内存屏障之前的所有写操作 都要写回到主内存
  • 内存屏障之后的所有读操作 都能获得内存屏障之前的所有写操作的最新结果(实现可见性)
写屏障(Store Memory Barrier): 告诉处理器在写屏障之前将所有存储在缓存(store bufferes)中的数据同步到主内存。也就是说当看到Store屏障指令,就必须把该指令之前所有写入指令执行完毕才能继续往下执行。
读屏障(Load Memory Barrier): 处理器在读屏障之后的读操作,都在读屏障之后执行。也就是说在Load屏障指令之后就能够保证后面的读取数据指令一定能够读取到最新的数据。

因此重排序时,不允许把内存屏障之后的指今重排序到内存屏障之前。一句话,对一个volatie变量的写,先行发生于任意后续对这个volatile变量的读,也叫写后读


内存屏障的分类

前文介绍的 happens-before 先行发生原则,类似接口规范,那么 我们如何对此规范进行落地实现?

实现它的 就是 内存屏障

内存屏障 可以分为 读屏障 (Load Barrier)写屏障(Store Barrier)

  • 读屏障 (Load Barrier)
    在读指令之前插入读屏障,让工作内存或CPU高速缓存当中的缓存数据失效,重新回到主内存中获取最新数据
  • 写屏障(Store Barrier)
    在写指令之后插入写屏障,强制把写缓冲区的数据刷回到主内存中

我们对这个屏障进行源码分析 看看它到底是什么样子的
源码下载连接如下:
openJDK8u


unsafe.class
在这里插入图片描述
在这里插入图片描述

OrderAccess.hpp
在这里插入图片描述

屏障的真实底层处理内容 是汇编语句
在这里插入图片描述

对二者进一步划分 可以划分为:

  • 读读屏障(LoadLoad)
  • 读写屏障(LoadStore)
  • 写写屏障(StoreStore)
  • 写读屏障(StoreLoad)
屏障类型指令类型说明
loadLoadLoad1;LoadLoad;Load2保证load1的读取操作在 load2及后续读取操作之前执行
StoreStoreStore1;storeStore;Store2在store2及其以后的写操作执行前,保证store1的写操作已经刷新到主内存
LoadStoreLoad;LoadStore;Store2在store2及以后的写操作执行前,保证load1的读操作已读取结束
StoreLoadStore1;StoreLoad;Loaded保证store1的写操作已经刷新到主内存之后,load2及以后的读操作才能执行

我们整理一下 关于JMM、重排序的思路

  1. 重排序有可能影响程序的执行和实现,因此,我们有时候希望告诉JVM你别“自作聪明”给我重排序,我这里不需要排序,听主人的。
  2. 对于编译器的重排序,JMM会根据重排序的规则,禁止特定类型的编译器重排序。(happen-before)
  3. 对于处理器的重排序,Java编译器在生成指令序列的适当位置,插入内存屏障指令,来禁止特定类型的处理器排序。

我们通过一个Demo 来更具象的理解:

/**
 * @Title:
 * @Description: TODO
 * @author: Alex
 * @Version:
 * @date 2023-02-17-0:43
 */
public class FenceDemo {

    volatile static int a = 1;

    public static void main(String[] args) {

        new Thread(() -> {

            while (true) {

                //第一个操作是volatile读
                System.out.println(Thread.currentThread().getName() + "\t" +
                        " get a value before every operations= " + a);


                if (a > 1) {
						//第三个操作是volatile读
                    System.out.println(Thread.currentThread().getName() + "\t" +
                            "second get a value = " + a); //load
                    break;
                }

            }
        }, "t1").start();


        new Thread(() -> {
            synchronized (FenceDemo.class) {
                //第二个操作是 volatile写  之间存在内存屏障 LoadStore 不能发生指令重排
                a++;  //store (存在间隙 注意非原子性)
            }
        }, "t2").start();

    }
}
 

输出结果如下:

t1	 get a value before every Load= 1
t1	 get a value before every Load= 1
t1	 get a value before every Load= 1
t1	 get a value before every Load= 1
t1	 get a value before every Load= 1
t1	 get a value before every Load= 1
t1	 get a value before every Load= 1
t1	 get a value before every Load= 1
t1	 get a value before every Load= 1
t1	 get a value before every Load= 1
t1	 get a value before every Load= 1
t1	 get a value before every Load= 1
t1	 get a value before every Load= 1
t1	 get a value before every Load= 1
t1	second get a value = 2

这个例子符合了我们预想的,

  1. 先读读看a 的值是 多少? load1
  2. 修改a的值 store2
  3. 判定a>1 再次去执行 读操作 load3

load 1       //volatile读操作
LoadStore      读写屏障 保证先读到初值,而不是被修改后的值(防止load1被重排)
store2      //volatile 写操作
StoreLoad     //写读屏障 防止后续读操作,读到未被正确修改的值
load3     //volatile读操作

形象的总结:
一锅汤,我先尝尝咸淡(load1) ,味道淡了,加点盐(store2), 再去常常咸淡(load3)

这样一系列操作,也是符合常理的,如果先加盐,再尝咸淡,那就离谱了。

下一步,我们加深案例难度:
如果出现普通读呢?

//TODO
01点34分 2023年2月17日

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

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

相关文章

万字讲解你写的代码是如何跑起来的?

今天我们来思考一个简单的问题&#xff0c;一个程序是如何在 Linux 上执行起来的&#xff1f; 我们就拿全宇宙最简单的 Hello World 程序来举例。 #include <stdio.h> int main() {printf("Hello, World!\n");return 0; } 我们在写完代码后&#xff0c;进行…

【THREE.JS学习(1)】绘制一个可以旋转、放缩的立方体

学习新技能&#xff0c;做一下笔记。在使用ThreeJS的时候&#xff0c;首先创建一个场景const scene new THREE.Scene();接着&#xff0c;创建一个相机其中&#xff0c;THREE.PerspectiveCamera&#xff08;&#xff09;四个参数分别为&#xff1a;1.fov 相机视锥体竖直方向视野…

算法拾遗二十六之暴力递归到动态规划五

算法拾遗二十五之暴力递归到动态规划五题目一&#xff08;返回K次打击后英雄把怪兽砍死的几率&#xff09;【样本对应模型&#xff0c;N和K是样本】题目二&#xff08;返回组成aim的最少货币数&#xff09;从左往右尝试模型题目三&#xff08;返回裂开的数的种类&#xff09;题…

【Kotlin】Kotlin函数那么多,你会几个?

目录标准函数letrunwithapplyalsotakeIftakeUnlessrepeat小结作用域函数的区别作用域函数使用场景简化函数尾递归函数&#xff08;tailrec&#xff09;扩展函数高阶函数内联函数&#xff08;inline&#xff09;inlinenoinlinecrossinline匿名函数标准函数 Kotlin标准库包含几个…

CUDA的统一内存

CUDA的统一内存 文章目录CUDA的统一内存N.1. Unified Memory IntroductionN.1.1. System RequirementsN.1.2. Simplifying GPU ProgrammingN.1.3. Data Migration and CoherencyN.1.4. GPU Memory OversubscriptionN.1.5. Multi-GPUN.1.6. System AllocatorN.1.7. Hardware Coh…

如何学习 Web3

在本文中&#xff0c;我将总结您可以采取的步骤来学习 Web3。从哪儿开始&#xff1f;当我们想要开始新事物时&#xff0c;我们需要一些指导&#xff0c;以免在一开始就卡住。但我们都是不同的&#xff0c;我们有不同的学习方式。这篇文章基于我学习 Web3 的非常个人的经验。路线…

SpringBoot集成WebSocket实现客户端与服务端长连接通信

场景&#xff1a; WebSocket协议是用于前后端长连接交互的技术&#xff0c;此技术多用于交互不断开的场景。特点是连接不间断、更轻量&#xff0c;只有在关闭浏览器窗口、或者关闭浏览器、或主动close&#xff0c;当前会话对象才会关闭。 这里只是简单的记录一下使用方式 一、服…

Proxy lab

CSAPP Proxy Lab 本实验需要实现一个web代理服务器&#xff0c;实现逐步从迭代到并发&#xff0c;到最终的具有缓存功能的并发代理服务器。 Web 代理是充当 Web 浏览器和终端服务器之间的中间人的程序。浏览器不是直接联系终端服务器获取网页&#xff0c;而是联系代理&#x…

关系型数据库的三大范式

一、简而言之 1、是什么&#xff1f; 三大范式是针对关系型数据库的一种数据库设计规范&#xff0c;使数据库设计符合约定的规范要求。 2、为什么要符合该规范&#xff1f; 为了建立冗余较小、结构合理的数据库。 3、三大范式内容的简单理解&#xff08;Normal Form&#…

2023美赛D题:可持续发展目标

以下内容全部来自人工翻译&#xff0c;仅供参考。 文章目录背景要求术语表文献服务背景 联合国制定了17个可持续发展目标&#xff08;SDGs&#xff09;。实现这些目标最终将改善世界上许多人的生活。这些目标并不相互独立&#xff0c;因此&#xff0c;一些目标的积极进展常常…

2023美国大学生数学建模竞赛选题建议

总的来说&#xff0c;这次算是美赛环境题元年&#xff0c;以往没有这么多环境题目&#xff0c;大部分题目都是开放度相当高的题目。C君认为的难度&#xff1a;D>C>AE>BF&#xff0c;开放度&#xff1a;DF>ABE>C。A题 遭受旱灾的植物群落这次A题为环境类题目&…

【技术分享】在RK3568上如何烧录MAC

本次我们使用的是触觉智能基于RK3568研发的IDO-EVB3568来给大家演示如何烧录MAC。 这款开发板拥有四核A55&#xff0c;主频高达2.0G&#xff0c;支持高达8GB高速LPDDR4&#xff0c;1T算力NPU &#xff0c;4K H.265硬解码&#xff0c;4K HDMI2.0显示输出&#xff0c;支持双通…

AMEPD SSD1680 调试记录

AMEPD Active Martix Electrophoretic Display&#xff0c;有源矩阵电泳显示屏。就是电纸书那种屏&#xff0c;调试效果使用感受和我的Kindle差不多。屏幕参数屏幕 IC为SSD1680122*250&#xff0c;单bit控制&#xff0c;1为白&#xff0c;0为黑逐行刷新&#xff0c;一个字节8bi…

JavaScript 浏览器中执行

本章节为大家介绍如何在浏览器上进行 JavaScript 代码的运行与调试。目前的主流浏览器有谷歌的Chrome&#xff08;使用blink内核&#xff09;&#xff0c;微软的edge&#xff08;使用chromium内核&#xff0c;这是一款谷歌提供的开源浏览器内核&#xff09;和IE&#xff08;使用…

记录锁,间隙锁,插入意向锁,临键锁兼容关系

插入意向锁是什么&#xff1f; 注意&#xff01;插入意向锁名字里虽然有意向锁这三个字&#xff0c;但是它并不是意向锁&#xff0c;它属于行级锁&#xff0c;是一种特殊的间隙锁。 在MySQL的官方文档中有以下重要描述&#xff1a; An Insert intention lock is a type of gap…

羊了个羊游戏开发教程3:卡牌拾取和消除

本文首发于微信公众号&#xff1a; 小蚂蚁教你做游戏。欢迎关注领取更多学习做游戏的原创教程资料&#xff0c;每天学点儿游戏开发知识。嗨&#xff01;大家好&#xff0c;我是小蚂蚁。终于要写第三篇教程了&#xff0c;中间拖的时间有点儿长&#xff0c;以至于我的好几位学员等…

2023美赛C题思路数据代码分享

文章目录赛题思路2023年美国大学生数学建模竞赛选题&论文一、关于选题二、关于论文格式三、关于论文提交四、论文提交流程注意不要手滑美赛C题思路数据代码【最新】赛题思路 (赛题出来以后第一时间在CSDN分享) 最新进度在文章最下方卡片&#xff0c;加入获取一手资源 202…

【小西】同步咪咕订单给咪咕方(写接口给第三方)

同步咪咕订单给咪咕方前言思路实现1、定义请求体和响应信息MiGuOrderSyncReqMiGuOrderSyncResp2、nacos定义好咪咕相关配置信息3、同步咪咕参数配置4、MiGuOrderSyncControl5、MiGuOrderSyncService6、MiGuOrderSyncServiceImplCreateAscIISignUtil 生成参数 字典排序 签名Hmac…

数据分析:消费者数据分析

数据分析&#xff1a;消费者数据分析 作者&#xff1a;AOAIYI 创作不易&#xff0c;如果觉得文章不错或能帮助到你学习&#xff0c;记得点赞收藏评论一下哦 文章目录数据分析&#xff1a;消费者数据分析一、前言二、数据准备三、数据预处理四、个体消费者分析五、用户消费行为总…

【CMake】CMake构建C++代码(一)

在Linux开发过程中&#xff0c;难免会用到CMake来构建你的代码。本文将说明如何构建自己的代码&#xff0c;将自己的代码变为共享库&#xff0c;共其他代码使用。 文章目录在Linux开发过程中&#xff0c;难免会用到CMake来构建你的代码。本文将说明如何构建自己的代码&#xff…