关于指令重排现象的两个阶段详解

目录
  • 编译期指令重排
    • 1、上神秘代码
    • 2、编译成Java字节码(没加volatile)
    • 3、编译成Java字节码(加了volatile)
    • 4、编译器优化
  • 运行期指令重排

那什么时候会产生指令重排现象呢?两个阶段: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修饰,上图。

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)

3、编译成Java字节码(加了volatile)

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

4、编译器优化

指令重排是编译器优化中的一种,编译openjdk是启用了O2级编译器优化,如图。

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

优化无效代码,看图(我就不贴C++代码了)

运行期指令重排

不知道大家有没有听过一个词:CPU乱序执行。乱序执行是相对于顺序执行来说的。计算机刚被发明的时候都是顺序执行,后来为了提升CPU运行效率,升级成了乱序执行。

那为什么乱序执行就提高了运行效率呢?有兴趣的童鞋可以去研究下,关键词:指令流水线。

所以计算机这行,如果你觉得大学学的那些基础知识不重要,你看我的文章就明白有多重要。这行走到最后较量的就是这些东西,就是看谁研究得更深入更底层更明了。

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

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

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

有些流程的顺序是可以提前知晓并确定下来,但有些流程的顺序是无法提前知晓的,比如你公司的业务,写JVM的人肯定不知道,所以依然需要程序猿根据业务需要来控制,那从JVM层面来说,我给你提供机制。内存屏障就是这种机制中的一种,其他的还有各种锁。关于内存屏障,我之前已经写了一篇文章深入讲解了这块,有兴趣的同学可以去看看,传送门 内存屏障由来及实现思路

以上就是关于指令重排现象的两个阶段详解的详细内容,更多关于指令重排现象的两个阶段的资料请关注我们其它相关文章!

(0)

相关推荐

  • Java中volatile防止指令重排

    目录 什么是指令重排? 为什么指令重排能够提高性能 volatile是怎么禁止指令重排的? volatile可以防止指令重排,在多线程环境下有时候我们需要使用volatile来防止指令重排,来保证代码运行后数据的准确性 什么是指令重排? 计算机在执行程序时,为了提高性能,编译器和处理器一般会进行指令重排,一般分为以下三种: 指令重排有以下三个特点: 1.单线程环境下指令重排后可以保证与顺序执行指令的结果一致(就是不进行指令重排的情况) //原来的执行顺序 a=1; b=0; //进行指令重排后执

  • Java volatile如何实现禁止指令重排

    计算机在执行程序时,为了提高性能,编译器和处理器常常会对指令重排,一般分为以下三种: 源代码 -> 编译器优化的重排 -> 指令并行的重排 -> 内存系统的重排 -> 最终执行指令 单线程环境里面确保最终执行结果和代码顺序的结果一致 处理器在进行重排序时,必须要考虑指令之间的数据依赖性 多线程环境中线程交替执行,由于编译器优化重排的存在,两个线程中使用的变量能否保证一致性是无法确定的,结果无法预测. 指令重排 - example 1 public void mySort() { i

  • 关于指令重排现象的两个阶段详解

    目录 编译期指令重排 1.上神秘代码 2.编译成Java字节码(没加volatile) 3.编译成Java字节码(加了volatile) 4.编译器优化 运行期指令重排 那什么时候会产生指令重排现象呢?两个阶段:1.编译期:2.运行期. 编译期指令重排 解释型语言是在运行期间执行编译+运行动作,所以运行效率较编译型语言低.Java既可以作为解释型语言去用,也可以作为编译型语言.但是主流的做法是当成编译型语言在用.那Java在编译期做了指令重排优化吗?做了哪些优化?能不能让我看看?为了满足大家的好

  • IOC 容器启动和Bean实例化两个阶段详解

    目录 IOC 容器的两个阶段 容器启动阶段 Bean 实例化阶段 插手容器的启动 PropertyPlaceholderConfigurer 占位符机制 PropertyOverrideConfigurer 重写属性值 实例-使用容器扩展点修改 BeanDefinition IOC 容器的两个阶段 IOC 容器可以分为两个阶段 : 容器启动阶段和 Bean 实例化阶段. Spring 的 IoC 容器在实现的时候, 充分运用了这两个阶段的不同特点, 在每个阶段都加入了相应的容器扩展点, 支持开发

  • RabbitMQ 实现延迟队列的两种方式详解

    目录 1. 用插件 1.1 安装插件 1.2 消息收发 2. DLX 实现延迟队列 2.1 延迟队列实现思路 2.2 案例 3. 小结 定时任务各种各样,常见的定时任务例如日志备份,我们可能在每天凌晨 3 点去备份,这种固定时间的定时任务我们一般采用 cron 表达式就能轻松的实现,还有一些比较特殊的定时任务,向大家看电影中的定时炸弹,3分钟后爆炸,这种定时任务就不太好用 cron 去描述,因为开始时间不确定,我们开发中有的时候也会遇到类似的需求,例如: 在电商项目中,当我们下单之后,一般需要

  • Python获取网络时间戳的两种方法详解

    目录 方法一 代码实现 调用方法 返回结果 方法二 代码实现 调用方法 返回结果 在我们进行注册码的有效期验证时,通常使用获取网络时间的方式来进行比对. 以下为获取网络时间的几种方式. 方法一 需要的时间会比较长,个别电脑上可能会出现不兼容现象 代码实现 def get_web_server_time(self, host_URL, year_str='-', time_str=':'): ''' 获取网络时间,需要的时间会比较长,个别电脑上可能会出现不兼容现象 :param host_URL:

  • Vue指令实现大屏元素分辨率适配详解

    目录 前言 1. 常见的适配方案 2. CSS3 缩放方案 3. 封装一个缩放指令 4. 后记 前言 随着前端技术的不断发展.数据中心(中台)之类的概念的不断升级.物联网设备的更新和普及,越来越多的业主(项目)喜欢在系统中添加一个或者多个可视化大屏,用来集中的展现数据变化.位置变化等等,老板们也更喜欢称之为“态势”. 当然,作为程序员一般都不关心“老板们”的想法,只要完成项目即可.但是经常会有这样的问题:我有一个大屏的模板,但是用户的浏览器分辨率不够,或者有的有书签栏有的没有书签栏,更或者是有的

  • C语言中 int main(int argc,char *argv[])的两个参数详解

    C语言中 int main(int argc,char *argv[])的两个参数详解 argc是命令行总的参数个数: argv[]是argc个参数,其中第0个参数是程序的全名,以后的参数.命令行后面跟的用户输入的参数. int main(int argc, char* argv[]) { int i; for (i = 0; i<argc; i++) cout<<argv[i]<<endl; cin>>i; return 0; } 执行时敲入 F:\MYDOCU

  • 基于ScheduledExecutorService的两种方法(详解)

    开发中,往往遇到另起线程执行其他代码的情况,用java定时任务接口ScheduledExecutorService来实现. ScheduledExecutorService是基于线程池设计的定时任务类,每个调度任务都会分配到线程池中的一个线程去执行,也就是说,任务是并发执行,互不影响. 注意,只有当调度任务来的时候,ScheduledExecutorService才会真正启动一个线程,其余时间ScheduledExecutorService都是处于轮询任务的状态. 1.scheduleAtFix

  • 使用Java构造和解析Json数据的两种方法(详解二)

    JSON(JavaScript Object Notation) 是一种轻量级的数据交换格式,采用完全独立于语言的文本格式,是理想的数据交换格式.同时,JSON是 JavaScript 原生格式,这意味着在 JavaScript 中处理 JSON数据不须要任何特殊的 API 或工具包. 在www.json.org上公布了很多JAVA下的json构造和解析工具,其中org.json和json-lib比较简单,两者使用上差不多但还是有些区别.下面接着介绍用org.json构造和解析Json数据的方法

  • IOS 指纹识别两种方式详解及实例

    IOS 指纹识别两种方式详解及实例 首先引入类名: #import <LocalAuthentication/LocalAuthentication.h> 然后在实现指纹识别的地方放入如下代码: 方式一: LAContext *lacontext = [[LAContext alloc]init]; // 判断设备是否支持指纹识别 BOOL isSupport = [lacontext canEvaluatePolicy:LAPolicyDeviceOwnerAuthenticationWit

  • 使用Java构造和解析Json数据的两种方法(详解一)

    JSON(JavaScript Object Notation) 是一种轻量级的数据交换格式,采用完全独立于语言的文本格式,是理想的数据交换格式.同时,JSON是 JavaScript 原生格式,这意味着在 JavaScript 中处理 JSON数据不须要任何特殊的 API 或工具包. 在www.json.org上公布了很多JAVA下的json构造和解析工具,其中org.json和json-lib比较简单,两者使用上差不多但还是有些区别.下面首先介绍用json-lib构造和解析Json数据的方法

随机推荐