编译器指令重排和CPU指令重排_十进制调整指令DA怎么用

编译器指令重排和CPU指令重排_十进制调整指令DA怎么用这个知识点也是很多人说不清道不明的地方,感觉都知道,说又说不出来。为什么会这样呢?因为这几个字,很容易被当成动词去理解,其实正确的理解是当成名词,即指令重排现象。那什么时候会产生指令重排现象呢?两个阶段:1、编译期;2、运行期。编译期指令重排解释型语言是在运行期间执行编译+运行动作,所以运行效率较编译型语言低。Java既可以作为解释型语言去用,也可以作为编译型语言。但是主流的做法是当成编译型语言在…

大家好,又见面了,我是你们的朋友全栈君。如果您正在找激活码,请点击查看最新教程,关注关注公众号 “全栈程序员社区” 获取激活教程,可能之前旧版本教程已经失效.最新Idea2022.1教程亲测有效,一键激活。

Jetbrains全系列IDE稳定放心使用

这个知识点也是很多人说不清道不明的地方,感觉都知道,说又说不出来。为什么会这样呢?因为这几个字,很容易被当成动词去理解,其实正确的理解是当成名词,即指令重排现象。那什么时候会产生指令重排现象呢?两个阶段:1、编译期;2、运行期。

编译期指令重排

解释型语言是在运行期间执行编译+运行动作,所以运行效率较编译型语言低。Java既可以作为解释型语言去用,也可以作为编译型语言。但是主流的做法是当成编译型语言在用。那Java在编译期做了指令重排优化吗?做了哪些优化?能不能让我看看?为了满足大家的好奇,安排。

这里先解释下编译期:像c/c++只有一个编译期,就是调用gcc命令将c/c++代码编译成汇编代码。但是Java中有两个编译期:1、调用javac命令将Java代码编译成Java字节码;2、Unix派系平台上调用gcc命令将openjdk源码编译成汇编代码。网上所有的文章都是在讲第一种,而且都是讲概念,以讹传讹。我这篇文章不仅两种都讲,还都用代码+图片的方式证明给你看。所以想学底层,不找一个靠谱的师傅是学不会学不明白的,因为第一你不知道这个知识点牵扯得有多深,第二两个观点摆在你面前,你不知道哪个对那个错。

这里我先把结论给大家吧:编译期间,Java中所谓的指令重排主要是说编译openjdk时的指令重排,将Java代码编译成Java字节码是没有做指令重排的。即你加不加volatile,生成的字节码文件是一样的。是不是颠覆了你对这块的认知呢!不信?看案例。

可能有人要问了,如果加不加volatile生成的字节码文件都一个样,那在运行的时候JVM是怎么知道的呢?类属性在JVM中存储的时候会有一个属性:Access flags。JVM在运行的时候就是通过该属性来判断操作的类属性有没有加volatile修饰,上图。编译器指令重排和CPU指令重排_十进制调整指令DA怎么用编译器指令重排和CPU指令重排_十进制调整指令DA怎么用

1、上神秘代码

public class Test3 {

public static /* volatile */ int found = 0;

public static void main(String[] args) {

new Thread(new Runnable() {

public void run() {

System.out.println(“等基友送笔来…”);

while (0 == found) {

}

System.out.println(“笔来了,开始写字…”);

}

}, “我线程”).start();

new Thread(new Runnable() {

public void run() {

try {

Thread.sleep(2000);

} catch (InterruptedException e) {

e.printStackTrace();

}

System.out.println(“基友找到笔了,送过去…”);

change();

}

}, “基友线程”).start();

}

public static void change() {

found = 1;

}

}

稍微解释下这段代码:有两个线程:我线程、基友线程。『我线程』通过死循环阻塞在那里等待『基友线程』找到笔送过来,然后开始写字。『基友线程』等待一会就去找笔,找到了就送过去。

2、编译成Java字节码(没加volatile)编译器指令重排和CPU指令重排_十进制调整指令DA怎么用

3、编译成Java字节码(加了volatile)编译器指令重排和CPU指令重排_十进制调整指令DA怎么用

可以发现加不加volatile,生成的字节码是一样的。

4、编译器优化

指令重排是编译器优化中的一种,编译openjdk是启用了O2级编译器优化,如图。编译器指令重排和CPU指令重排_十进制调整指令DA怎么用

O2级优化做了哪些优化?比如优化无效代码、编译期完成简单运算、处理编译期屏障……那gcc有多少级优化?有兴趣的童鞋可以自行学习,百度搜索关键词:-O2。

优化无效代码,看图(我就不贴C++代码了)编译器指令重排和CPU指令重排_十进制调整指令DA怎么用

运行期指令重排

不知道大家有没有听过一个词:CPU乱序执行。乱序执行是相对于顺序执行来说的。计算机刚被发明的时候都是顺序执行,后来为了提升CPU运行效率,升级成了乱序执行。那为什么乱序执行就提高了运行效率呢?有兴趣的童鞋可以去研究下,关键词:指令流水线。所以计算机这行,如果你觉得大学学的那些基础知识不重要,你看我的文章就明白有多重要。这行走到最后较量的就是这些东西,就是看谁研究得更深入更底层更明了。

因为现在的CPU都是采用乱序执行,这样在运行程序的过程中就带来了指令重排的现象。这是在运行期,在CPU内部发生的,我就没办法证明给你看了。但就算是乱序执行提高了效率,那也不能改变我程序的意愿,这就引出了一个概念:as-if-serial。

何谓as-if-serial呢?简单的说就是不管你在编译期或者在运行期怎么做指令重排,单线程环境下程序的执行结果不能改变。说白了这是指令重排的底线,是必须遵守的规范。那如何保证呢?这就引出了另外两个难以理解的知识点:happens-before、内存屏障。

happens-before是做什么的呢?简单的说就是告诉写JVM的人,你写JVM的时候要遵循这几条规则,这几条规则是你JVM默认要做到的,而不用程序猿在写代码的时候需要去想去做控制。比如对象的初始化动作一定要先于finalize方法执行前完成。其他几个规则我就不细说了,都很好理解,童鞋们自行去学习下。

有些流程的顺序是可以提前知晓并确定下来,但有些流程的顺序是无法提前知晓的,比如你公司的业务,写JVM的人肯定不知道,所以依然需要程序猿根据业务需要来控制,那从JVM层面来说,我给你提供机制。内存屏障就是这种机制中的一种,其他的还有各种锁。关于内存屏障,我之前已经写了一篇文章深入讲解了这块,有兴趣的同学可以去看看。子牙老师:聊聊内存屏障​zhuanlan.zhihu.com编译器指令重排和CPU指令重排_十进制调整指令DA怎么用

至此,指令重排就算讲明白了,不晓得童鞋们有么有看明白、理解到位。

子牙老师喜欢分析硬核文章,都会在个人公众号(启明南)首发。想第一时间获得通知吗?那就关注一波吧。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/184346.html原文链接:https://javaforall.cn

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)


相关推荐

  • JAVA字符串格式化-String.format()的使用

    JAVA字符串格式化-String.format()的使用JAVA字符串格式化-String.format()的使用

  • Quartz任务中调用Spring容器中bean及动态调度任务-SchedulerFactoryBean「建议收藏」

    Quartz任务中调用Spring容器中bean及动态调度任务-SchedulerFactoryBean「建议收藏」Quartz是开源任务调度框架中的翘首,它提供了强大任务调度机制,同时保持了使用的简单性。Quartz允许开发人员灵活地定义触发器的调度时间表,并可以对触发器和任务进行关联映射。此外,Quartz提供了调度运行环境的持久化机制,可以保存并恢复调度现场,即使系统因故障关闭,任务调度现场数据并不会丢失。此外,Quartz还提供了组件式的侦听器、各种插件、线程池等功能。Spring为…

  • linux平均负载什么意思_linux服务器负载高

    linux平均负载什么意思_linux服务器负载高1,Linux系统的平均负载是什么?特定时间间隔内运行队列中的平均进程数,好象还不够明白:就是进程队列的长度,有多少个进程在排队等待运行2,什么是”进程队列”?一个进程满足以下条件就会位于进程队列中1,它没有在等待I/O操作的结果2,它没有主动进入等待状态(即没有调用wait)3,它没有被停止3,如何查看平均负载?最简单的命令是uptime例子:[www.linuxidc.com@localho…

  • 解决虚拟机选择桥接模式连不上网

    解决虚拟机选择桥接模式连不上网虚拟机装的centos7,使用桥接模式,且与主机在同一网段,主机与虚拟机可以互相ping通,且主机使用secureCRT可以连接虚拟机。主机可以上网,但是虚拟机里centos就是连不上网,以前都可以上网。再查了一堆资料都没解决后,总算找到一篇解决办法。原文:原文地址保证你Vmware里面的虚拟机是关机状态 右键点击电脑屏幕右下角小电脑图标,选择打开网络与共享中心,然后点…

  • vscode搭建react框架(vscode补全js方法)

    文章目录一、安装node二、配置淘宝镜像三、配置vscode(win10)四、全局安装脚手架五、创建项目一、安装node请在官网下载安装:https://nodejs.org/zh-cn/vscode中点击(ctrl+`)调出终端输入指令node-v,能显示版本号,说明node已经装好了输入指令npm-v,能显示版本号,说明npm可以使用了点击链接查看图文教程https://blog.csdn.net/qq_45677671/article/detail

  • Android最常用的控件ListView(详解)

    Android最常用的控件ListView(详解)一.ListView简介在Android开发中,ListView是一个比较常用的控件。它以列表的形式展示具体数据内容,并且能够根据数据的长度自适应屏幕显示。二.ListView简单用法代码部分1.布局界面activity_main.xml代码:<?xmlversion=”1.0″encoding=”utf-8″?><LinearLayoutxmlns:android=”http://schemas.android.com/ap…

发表回复

您的电子邮箱地址不会被公开。

关注全栈程序员社区公众号