内存屏障,先看这篇文章

刚看到这个词的时候,我以为是白内障,然后查了很多资料,才理解内存屏障是干嘛的,我就不像很多其他文章说得那么多了,我希望我说得简单一些,让大家看了我的文章都知道这个是怎么回事。

说到内存屏障,我们先从CPU性能优化说起

性能优化的方法一,缓存

CPU的速度很快,到底有多快?我们就用光速来比喻CPU的执行速度吧,反正就是执行读写很快,但是CPU速度很快,内存的速度很慢,怎么办?

这时候,高速缓存出现了「先不抬杠说寄存器哈」。比如有一个变量a ,CPU在很长一段时间内都需要使用它,如果把他放在内存里面的话,每次读写的速度都很慢,这样严重拖慢了CPU的执行速度。

所以就出现了 缓存

缓存也是分类的,缓存分成了三类,我简单说明下

缓存同步协议MESI如果在4个CPU在各自的L1 里面都有一个变量 i ,他们要把这个变量i写入到内存里面去,是以哪一个为准呢?

这个时候就需要缓存的同步协议,每个cpu对缓存里面的变量操作的时候,要通知其他CPU,让他们知道当前的状态。

CPU不仅需要发消息给其他CPU告诉他们状态,同时也要接收其他CPU发出来的状态。

这样做的最终目的,就是为了保证CPU对变量操作的一致性

CPU性能优化的方法二,运行时指令重排

指令重排是个有意思的事情,如果把CPU当成是一个人的话,他也是有自己的意识的,存在意识的东西,就会存在自己做事情的方法,比如,我今天要去打篮球,还要去帮妈妈打酱油,我是先打篮球还是先打酱油,因为个人的意识不同,做事情的先后顺序也存在差异。

为什么出现指令重排?如果有一个CPU,我们假设是CPU0吧,它在写缓存的的某个区块的时候,发现这个缓存位置刚好被CPU1正常操作,那他怎么办?有两种方法,一种是等待CPU1执行结束后自己再执行,还有一种方法就是先去干其他的事情,很明显,CPU为了提高自己的性能,它会选择第二种方法,就是先去干其他的事情,干其他的事情,就出现上面描述的情况,指令重排了。

所以我们会有一个疑问?代码是按照程序员的想法去执行的呢?还是按照CPU的想法去执行的呢?回答当然是需要按照程序员的意识去执行的。

所以要求,指令重排遵循as-if-serial语义这个是什么意思呢?就是说指令重排的结果不能影响程序员预期的结果,比如上面的代码,重排后就会出现问题,那么CPU就不能对指令进行重排。

注意,上面所说的as-if-serial语义针对的是单核CPU来说

但是如果是下面的代码

a = 100;
b = c;

上面两条语句不存在依赖关系,它们可以进行指令重排,因为重排后不会影响最后的执行结果。

高速缓存和指令重排序(reordings)存在的问题

缓存机制和指令重排都是为了CPU运算的性能优化。会出现两个问题。1、缓存和内存的数据并不是时实同步的,同一个内存地址,不同的CPU看到的内存值是不一样的。因为CPU运行速度很快,假设,CPU0 读 地址 0x2345 的时候值为 1,这个时候,CPU1向 0x2345写入了 2,CPU2再读这个内存 0x2345 的值的时候,发现它变成2了。所以就有问题了

出现的问题是同一个时间点上,不同的CPU看到的同一个内存地址数据不一样

2、我们指令重排说的as-if-serial语义只是针对单个CPU,那多个CPU呢?会出现什么问题

看下面的例子:

看上面图片

CPU0 要执行两条指令,CPU1也要执行两条指令,但是如果他们执行的先后顺序不同,那么x和y的结果也将存在差异。

第一种情况如下图执行顺序会导致 x = 0 , y = 1。

第二种情况如下图执行顺序会导致 x =1,y = 0。第三种情况如下图执行顺序会导致 x = 1,y=1。

结果跟程序员的预期不符合 出现的问题是,多个CPU,也就是我们经常所说的SMP系统下,会出现结果和预期不一致的问题

最后说内存屏障(Memory Barrier)

内存屏障就是用来解决上面两个问题的。这个是CPU厂商来搞定的。

  • 写内存屏障(Store Memory Barrier) 写内存屏障的意思就是在写内存的后面加入指令 Store Barrier ,如果CPU有读的也有写的,加了这条指令,就保证先执行写入而不去做指令重排,这种显示调用可以让其他线程看到。其他线程会等这个执行结束后再去操作,既然是等待,那也是降低性能的,好吧,降低性能也是没有办法的事情了。

就拿上面的 图片来说明A = 1 B = 1 这两个是写操作,我们加上写内存屏障,即使其他CPU有读的指令,我们需要等待这个写完成后,再进行读操作。

  • 读内存屏障 (Load Memory Barrier) 在读指令之前插入Load Barrier,可以让高速缓存中的数据失效,强制从内存加载最新的数据,让CPU缓存和主内存保持一致,避免了缓存导致的一致性问题。

void executedOnCpu0() {value = 10;/*在更新数据之前必须将所有存储缓存(store buffer)中的指令执行完毕。*/storeMemoryBarrier();finished = true;
}
void executedOnCpu1() {while(!finished);/*在读取之前将所有失效队列中关于该数据的指令执行完毕。*/loadMemoryBarrier();assert value == 10;
}

总结

CPU为了性能,发明了缓存和指令重排,但是又因为缓存和指令重排出现了新的问题,因为新的问题出现,聪明的人类又发明了内存屏障,之所谓,兵来将挡水来土掩就是这个道理。

文章是整理了自己看到资料的很多见解,后续会发新的文章进一步讲解,当然了,或者也会不发,我就是一只漂亮的鸽子,我鸽呀鸽呀鸽,祝大家周末愉快。


扫码或长按关注

回复「 篮球的大肚子」进入技术群聊

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

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

相关文章

SpringMVC原理及非注解配置详解

如需转发请标明出处:http://www.cnblogs.com/gudu1/p/7222556.html 1. Spring介绍 Spring MVC是Spring提供的一个强大而灵活的web框架。借助于注解,Spring MVC提供了几乎是POJO的开发模式,使得控制器的开发和测试更加简单。 这些控制器一般不…

ftp 追加远程文件_远程办公彻底火了,高效办公,拒绝卡顿,远程利器你选对了么?...

远程办公的同时肯定会应用到公司的内部系统了,比如OA、ERP等系统,当然为了提高远程办公人员的工作效率,实现资源共享最大化,企业FTP服务的应用也是必不可少。无论是员工出差还是修改,一旦涉及文件的应用,通…

android jxl.jar 使用,使用jxl.jar在Android中操作Excel表格——重中之重——对隐藏表的处理...

曾简单了解过C#,将Excel(数据库表)表中的数据导入到C#中,使用C#制作的图形化界面进行对Excel表中数据进行操作。今天想试试,在Android中导入Excel表格进行操作。在网上查阅资料,找到了jxl.jar包。jxl.jarjxl.jar是通过java操作exc…

自定义协议的这些典型例子你会了吗?

上次分享的《分享一个很酷的上位机软件》中,有如下协议:有位读者朋友问数据为什么要按这样的格式来发。其实这是个自定义协议,这是上位机开发者定义的一个数据交互协议。我们下位机往伏特加上位机发送数据需要遵循这样的协议数据,…

java 原子类_小学妹教你并发编程的三大特性:原子性、可见性、有序性

在并发编程中有三个非常重要的特性:原子性、有序性,、可见性,学妹发现你对它们不是很了解,她很着急,因为理解这三个特性对于能够正确地开发高并发程序有很大的帮助,接下来的面试中也极有可能被问到,小学妹就…

关于a标签不能调用js方法的小细节,你注意到了么?

在我们做后台删除的时候&#xff0c;当点击删除标签时&#xff0c;你希望弹出一个友好的提示框&#xff01;比如这样&#xff1a; 那代码应该怎样写呢&#xff1f;向下面这样&#xff1f; <!DOCTYPE html> <html lang"en"> <head><meta charset…

4岁小女孩给Linux内核贡献提交

今天在reddit上看到一个有趣的讨论&#xff0c;一个4岁的小女孩给Linux提交了一个补丁&#xff0c;并且这个补丁合并到了代码中。链接如下&#xff1a;https://www.reddit.com/r/linux/comments/2pqqla/kernel_commit_4_year_old_girl_fixes_formatting_to/cmzfvpl/我们看看她修…

php文件上传后没有打开权限_记墨者靶机文件上传(二)

“ 声明&#xff1a;该公众号大部分文章来自日常学习笔记&#xff0c;若是转载会先得到原作者授权或其他公众号白名单&#xff0c;并附上链接。剑者&#xff0c;心之刃也。既可为杀&#xff0c;亦可为护。杀与护&#xff0c;不过一念之间&#xff01;请勿利用文章内的相关技术从…

你还会写这段C51程序吗?

经典题目解析定时器T1采用计数模式&#xff0c;方式1中断&#xff0c;计数输入引脚 P3.5外接开关按钮作为计数信号输入&#xff0c;P1口控制8个LED小灯&#xff0c;初始状态所有小灯全亮&#xff0c;按3次按钮开关产生计数中断时&#xff0c;高3位和低5位交替闪烁3次&#xff0…

JDBC连接数据库(Java DataBase Connectivity,java)

通过JDBC操作数据库(以mysql为例) 1、官网下载JDBC的相关JAR包 【https://dev.mysql.com/downloads/connector/j/】 2、解压后&#xff0c;导入jar包 连接数据(引入外部jar包)项目上右键->build path->configure build path->上面Libraries->Add External JARs 3、…

过年回家抢票攻略

每年过年抢票都是一个困难的事情&#xff0c;所以决定发一篇文章汇总一下各种抢票软件&#xff0c;大家自己有方案的可以在文章后面留言&#xff0c;让跟多的人看到&#xff0c;希望大家都能顺利抢到车票&#xff0c;顺利回家。超级抢票机&#xff0c;一款超强的火车票抢票神器…

C语言,函数不可返回指向栈内存的指针

预备知识&#xff1a;内存的分类C/C程序占用的内存分为两大类&#xff1a;静态存储区与动态存储区。其示意图如下所示&#xff1a;数据保存在静态存储区与动态存储区的区别就是&#xff1a;静态存储区在编译-链接阶段已经确定了&#xff0c;程序运行过程中不会变化&#xff0c;…

android 界面绘制完毕,几种获取android 界面性能数据的快捷方法

探测 界面绘制性能获取界面的绘制性能有很多种方法&#xff0c;比如说 Systrace 但是这种方法 有一个不太好的地方就是使用起来较为复杂&#xff0c; 有没有一种 谷歌官方推荐 的方便一点的方法 &#xff0c;其实是有的&#xff0c;只需要一个函数 就可以获得layout的时间 非常…

联想rd650怎么装系统win7_Lenovo g50重装win7系统|U盘重装联想g50笔记本系统

Lenovo可以说是个家喻户晓的品牌很多小伙伴们也是使用联想的电脑&#xff0c;今天有小伙伴和小编提了一款Lenovo g50笔记本&#xff0c;小编了解到这款笔记本电脑的口碑和销量都很不错&#xff0c;其实这位小伙伴的问题呢是要如何重装win7系统&#xff0c;毕竟质量再好的电脑也…

书籍推荐

今天天气很好&#xff0c;风和日丽&#xff0c;艳阳高照&#xff0c;大家心情应该也很不错&#xff0c;毕竟&#xff0c;今天是周五&#xff0c;就像上学的时候一样&#xff0c;下午的铃声一响&#xff0c;每个同学都像脱缰的野马一样&#xff0c;周五&#xff0c;上班族的人们…

【转】JMeter学习(十三)分布式部署

Jmeter 是Java 应用&#xff0c;对于CPU和内存的消耗比较大&#xff0c;因此&#xff0c;当需要模拟数以千计的并发用户时&#xff0c;使用单台机器模拟所有的并发用户就有些力不从心&#xff0c;甚至会引起JAVA内存溢出错误。为了让jmeter工具提供更大的负载能力&#xff0c;j…

android mvp框架基类,Android MVP架构模式基类封装

前言MVP模式是Android官方推荐的架构模式&#xff0c;可使视图与数据层完全解耦。本文旨意封装在MVP模式中的基类如Activity&#xff0c;Fragment&#xff0c;Presenter类。以下内容建议在了解了mvp模式的读者阅读&#xff0c;如果还有对mvp架构模式有疑问的&#xff0c;请看我…

C语言,字符串指针做函数参数

看一下下面这段代码有什么问题&#xff1f;#include "stdio.h" //#include "stdbool.h" #include "string.h" #include "stdlib.h" #include "math.h"void getMemory(char *p) {/*char *p str*/p (char *)malloc(100);str…

java创建一个未知长度的数组_Java数组的创建操作

数组是一个固定长度的&#xff0c;包含了相同类型数据的 容器步骤1:声明数组步骤2:创建数组步骤3:访问数组步骤4:数组长度步骤5:练习-数组最小值步骤6:答案-数组最小值步骤 1 : 声明数组int[] a; 声明了一个数组变量。[]表示该变量是一个数组int 表示数组里的每一个元素都是一个…

Linux内核中的GPIO系统

一、前言作为一个工作多年的系统工程师&#xff0c;免不了做两件事情&#xff1a;培训新员工和给新员工分配任务。对于那些刚刚从学校出来的学生&#xff0c;一般在开始的时候总是分配一些非常简单的任务&#xff0c;例如GPIO driver、LED driver。往往CPU datasheet的关于GPIO…