深入理解Java虚拟机(九)——后端编译与优化
即时编译器
Java程序最初都是通过解释器进行执行,当发现某个方法或者代码块被运行得非常频繁,这些代码就被认为是热点代码,为了提高这些代码得运行效率,虚拟机会把热点代码编译成本地机器码,并进行优化,运行时完成这个任务的后端编译器被称为即时编译器。
解释器与编译器
主流Java虚拟机内部同时包含解释器与编译器。
解释器优点:当程序需要迅速启动和运行的时候,解释器可以省去编译的时间,立即运行代码。
编译器优点:当程序启动后,编译器将执行频繁的代码编译成本地代码,减少解释器的中间损耗,提高执行效率。
内存问题:编译器编译好的代码会占用本地内存,所以,如果内存不够,可以尽可能地利用解释编译,进行逆优化。
编译对象和触发条件
热点代码
- 被多次调用的方法。
- 被多次执行的循环体。
热点代码的编译目标都是整个方法,而不是单独的一段代码。
编译时会传入方法的入口字节码序号,然后在编译后,直接替换字节码初的方法,这种编译过程被称为栈上替换。
热点探测
检测某段代码是不是热点代码的行为被称为热点探测。
主流的两种热点探测方法:
- 基于采样的热点探测:周期性地去检查各个线程地调用栈顶,如果发现某个方法经常出现在栈顶,那么这个方法就是热点方法。
优点:简单高效,容易去获取方法调用关系。
缺点:很难精确计算一个方法的热度,容易受到干扰,如线程阻塞。 - 基于计数器地热点探测:虚拟机为每个方法建立一个计数器,统计方法地调用次数,经常被调用就是热点代码。
优点:精确计算每个方法的热度。
缺点:额外的开销维护计数器,花费空间,不能获取方法的调用关系。
HotSpot使用计数器的方法,并设计了两种计数器:
- 方法调用计数器:计算方法的调用次数。
- 回边计数器:计算循环代码的次数。
编译过程
客户端编译器
三段式编译:
- 第一个阶段,将子节码转化成一种高级的中间代码表示。目的时为了更用以实现代码的优化。也完成了一部分基础的优化,如方法内联和传播优化。
- 第二个阶段,再从上一个阶段的代码转化为低级中间代码表示,完成空值检查消除、范围检查消除等。
- 最后阶段,使用线性扫描算法,为上一步产生的代码分配寄存器,并做窥孔优化,然后产生机器码。

即时编译优点
- 性能分析制导优化:分析代码的运行的情况,进行集中优化。
- 激进预测性优化:虚拟机会根据继承类关系分析等一系列激进的猜测去做虚拟化,预测程序之后的运行情况,再优化。
- 链接优化:主程序和各个动态链接库可以各自进行优化。
提前编译器
破坏了Java语言的平台无关性。但是在Android平台上很有优势。
提前编译优点
- 不需要在运行的时候占用资源。
- 本质是给即时编译器做了缓存加速,改善程序的启动时间。
- 提前编译的时候没有执行时间和资源限制的压力,可以没有顾忌地进行优化。
编译器优化技术
方法内联
- 基本原理:将目标方法的代码转移到发起调用的地方,减少真实的方法调用。
需要通过类型继承关系分析,确认到底调用的是哪个方法。
逃逸分析
不是直接优化代码,而是一种为优化提供的分析技术。
- 基本原理:一个对象再方法中被定义后,如果被外部方法调用,就是方法逃逸;如果被外部线程访问,就是线程逃逸。不逃逸、方法逃逸、线程逃逸,就是对象从低到高的逃逸程度。根据对象的逃逸程度再进行不同的优化手段。
优化方法:
- 栈上分配:当确定一个对象不会被其他线程访问,就可以将对象分配到栈上,而不是堆上,这样可以减轻堆上垃圾回收的压力,让对象随着方法调用结束被销毁。支持方法逃逸,不支持线程逃逸。
- 标量替换:如果确定一个对象不会在方法外被访问,就可以在实例化对象的时候,不创建对象,而是拆散成多个被这个方法调用成员变量来代替。不支持方法逃逸和线程逃逸。
- 同步消除:如果确定一个变量不会回被其他线程访问,就可以消除对这个变量的同步措施,提高运行效率。
公共子表达式消除
- 基本原理:如果一个计算表达式之前被计算过了,而且其中的变量没有被修改,那就称为公共子表达式。对于这种表达式就不在计算,直接用之前计算过的结果进行代替。
数组边界检查消除
在编译的时候就判断数组是否会越界,这样在运行的时候就不需要每次在读取数组值的时候进行越界判断了。
示例
- 初始代码
static class B {
int value;
final int get() {
return value;
}
}
public void foo() {
y = b.get();
// ...do stuff...
z = b.get();
sum = y + z;
}
- 方法内联
public void foo() {
y = b.value;
// ...do stuff...
z = b.value;
sum = y + z;
}
- 冗余存储消除
public void foo() {
y = b.value;y
// ...do stuff...
z = y;
sum = y + z;
}
- 复写传播
public void foo() {
y = b.value;y
// ...do stuff...
y = y;
sum = y + y;
}
4.无用代码消除
public void foo() {
y = b.value;
// ...do stuff...
sum = y + y;
}
深入理解Java虚拟机(九)——后端编译与优化的更多相关文章
- 深入理解Java虚拟机(程序编译与代码优化)
文章首发于微信公众号:BaronTalk,欢迎关注! 对于性能和效率的追求一直是程序开发中永恒不变的宗旨,除了我们自己在编码过程中要充分考虑代码的性能和效率,虚拟机在编译阶段也会对代码进行优化.本文就 ...
- 深入理解Java虚拟机之自己编译JDK
题外话 最近在阅读<深入理解Java虚拟机>,其中有一小节实战是自己编译JDK,实际操作下来后遇到问题不少,为此特地记录,也希望可以给大家带来一些参考! 前置准备 平台及工具:Window ...
- 深入理解Java虚拟机 #01# 自己编译JDK
x 首先用书上的脚本尝试,失败. 之后根据源文件的 README 编译,抛出: root@linux:/opt/openjdk# sh ./get_source.sh ERROR: Need init ...
- 《深入理解java虚拟机》学习笔记之虚拟机即时编译详解
郑重声明:本片博客是学习<深入理解java虚拟机>一书所记录的笔记,内容基本为书中知识. Java程序最初是通过解释器(Interpreter)进行解释执行的,当虚拟机发现某个方法或代码块 ...
- 《深入理解java虚拟机》学习笔记之编译优化技术
郑重声明:本片博客是学习<深入理解Java虚拟机>一书所记录的笔记,内容基本为书中知识. Java程序员有一个共识,以编译方式执行本地代码比解释方式更快,之所以有这样的共识,除去虚拟机解释 ...
- 深入理解Java虚拟机--下
深入理解Java虚拟机--下 参考:https://www.zybuluo.com/jewes/note/57352 第10章 早期(编译期)优化 10.1 概述 Java语言的"编译期&q ...
- 《深入理解 Java 虚拟机》笔记整理
正文 一.Java 内存区域与内存溢出异常 1.运行时数据区域 程序计数器:当前线程所执行的字节码的行号指示器.线程私有. Java 虚拟机栈:Java 方法执行的内存模型.线程私有. 本地方法栈:N ...
- 《深入理解Java虚拟机》第 3 版里面到底多了哪些知识点?本文竟然得到了本书作者的认可!
这是why的第 47 篇原创文章 荒腔走板 大家好,我是 why.老规矩,先是简短的荒腔走板聊聊生活. 上面的图是前几天拍的,那天晚上下班后,刚刚走进小区就看到了这一轮弯月和旁边那一颗特别特别亮的星星 ...
- JVM | 第1部分:自动内存管理与性能调优《深入理解 Java 虚拟机》
目录 前言 1. 自动内存管理 1.1 JVM运行时数据区 1.2 Java 内存结构 1.3 HotSpot 虚拟机创建对象 1.4 HotSpot 虚拟机的对象内存布局 1.5 访问对象 2. 垃 ...
随机推荐
- JLC PCB 嘉立创自动确认生产稿,不讲武德?耗子尾汁!!!
首先,开局一张图,嘉立创又不做人的一天.嘉立创不讲武德,耗子尾汁!!! 之前下单,勾选了确定生产稿和不加客编,结果生产稿出来还是给我加了客编.那我出10元的意思何在?让我自己花3元看我花的10元有没有 ...
- Python基础数据类型与for循环
数据类型:int,bool,str,list, tuple元组,dict字典. 1.数字:12,3,4 在使用print打印数字时,在终端界面中无法判断出打印的是什么类型,当我们需要知道一个值是什么类 ...
- msfconsle核心命令学习
back 取消当前模块 banner check 检查当前exploit是否对目标有效,并不进行真正的攻击 color 禁用或启用输出是否包含颜色 connect 可以通过connect命令来链接Ne ...
- Camtasia中对录制视频进行编辑——交互性
随着新媒体的广泛发展,视频处理的需要也逐渐变得越来越大,很多人都不知道市场上的哪款软件是比较符合大众需要的.有的软件功能写的天花乱坠,但是实际操作确很难.并不符合大众的简单需求. 今天我便给大家推荐一 ...
- SQL注入步骤
1.判断是否存在注入,注入是字符型还是数字型2.猜解SQL查询语句中的字段数3.确定回显的字段数4.获取当前数据库5.获取表中字段名6.下载数据
- python之汉诺塔
# -*- coding: utf-8 -*- def move(n, a, b, c): if n==1: print(a,'==>',c)#只有一块的时候直接从A到C即可 else: mov ...
- 【刷题笔记】DP优化-状压
因为篇幅太长翻着麻烦,计划把DP拆成几个小专题,这里原文只留下状压,其他请至后续博文. 状态压缩优化 所谓状态压缩,就是将原本需要很多很多维来描述,甚至暴力根本描述不清的状态压缩成一维来描述. 时间复 ...
- java47
1. 1.List集合根据角标获取元素 import java.util.ArrayList; import java.util.List; public class List集合 { @Suppre ...
- B. Irreducible Anagrams【CF 1290B】
思路: 设tx为t类别字符的个数. ①对于长度小于2的t明显是"YES"②对于字符类别只有1个的t明显是"YES"③对于字符类别有2个的t,如左上图:如果str ...
- JZOJ2020年8月11日提高组T4 景点中心
JZOJ2020年8月11日提高组T4 景点中心 题目 Description 话说宁波市的中小学生在镇海中学参加计算机程序设计比赛,比赛之余,他们在镇海中学的各个景点参观.镇海中学共有n个景点,每个 ...