深入理解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. 垃 ...
随机推荐
- 信号发送接收函数:sigqueue/sigaction
信号是一种古老的进程间通信方式,下面的例子利用sigqueue发送信号并附带数据:sigaction函数接受信号并且处理时接受数据. 1.sigqueue: 新的信号发送函数,比kill()函数传递了 ...
- makefile 函数
1. findstring $(findstring <find>, <in>) 从 in 中查找 find ,如果找到则返回find,否则返回空 str1=1111 str2 ...
- yum安装Ceph指定Jewel版本
前言 通过yum安装指定的rpm包,这个一般是 yum --showduplicates list ceph | expand ,然后去通过yum安装指定的版本即可,这个在hammer下是没有问题的, ...
- 预估Ceph集群恢复时间
一.前言 本章很简单,就是预估集群恢复的时间,这个地方是简单的通过计算来预估需要恢复的实际,动态的显示 二.代码 #!/usr/bin/env python # -*- coding: UTF-8 - ...
- ubuntu配置简单的DNS服务器
之所以说是简单的服务器,实现的功能很简单,通过这个dns server 查询制定域名的时候,能够根据设置的值来返回IP,当前的需求是需要轮询的返回IP DNS 轮询机制会受到多方面的影响,如:A记录的 ...
- SQL SERVER 数据库自动备份及定期删除设置步骤
现在任何的软件都离不了一个数据库,数据的利用价值越来越大,为了避免数据宕机造成的数据丢失情况的产生,定期对数据库进行备份是必须要做的工作,下面将介绍SQL Server自带的数据库备份方法,希望可以帮 ...
- 重新认识C++的"cin >>"、"cout <<" 简简单单 - 快快乐乐
重新认识C++的"cin >>"."cout <<" 简简单单 - 快快乐乐 JERRY_Z. ~ 2020 / 11 / 24 转载请 ...
- MathType总结编辑括号的类型(上)
括号的种类有很多,我们用得也很多,可以说无处不见,不只是在数学物理这些自然科学的公式中来断地出现括号,即使是在人文艺术类的领域也会有括号的出现.下面就和小编一起来看看公式编辑器编辑括号的类型吧! Ma ...
- ABBYY FineReader 15 PDF文档编辑功能详解
ABBYY FineReader 15(Windows系统)OCR文字识别软件作为一款通用 PDF 工具,能轻松有效地对各种 PDF文档和纸质文档,进行数字化.检索.编辑.转换.包含.分享和合作,而其 ...
- Camtasia中如何自定义视频输出格式
Camtasia Studio是一款功能全面.操作简单的视频录制和编辑软件,它是很多需要进行录屏操作,比如制作教学视频的用户的不错选择.Camtasia 2020还为用户提供了极大的便利的全面的服务, ...