深入理解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. 垃 ...
随机推荐
- TypeError: Cannot read property 'Component' of undefined
继续跟着阮一峰的教程走,下面写到PropTypes的getDefaultProps时,又出现了问题,基于上一个createClass的报错换成了Component写法 错误描述: 解决方法:引入rea ...
- ubuntu服务器dns重启失效问题
方法一 通过/etc/network/interfaces,在它的最后增加一句: dns-nameservers 8.8.8.8 8.8.8.8是Google提供的DNS服务,这里只是举一个例子,你也 ...
- HotSpot类模型之InstanceKlass
上一篇 HotSpot源码分析之类模型 介绍了类模型的基础类Klass的重要属性及方法,这一篇介绍一下InstanceKlass及InstanceKlass的子类. 1.InstanceKlass类 ...
- flink1.10版本StreamGraph生成过程分析
1.StreamGraph本质 本质就是按照用程序代码的执行顺序构建出来的用于向执行环境传输的流式图,并且可以支持可视化展示给用户的一种数据结构. 2.StreamGraph.StreamNode和S ...
- day01-网络基础
一.知识点 1.socket.socket 创建一个 socket,该函数带有两个参数: Address Family:可以选择 AF_INET(用于 Internet 进程间通信) 或者 AF_UN ...
- python-网络安全编程第五天(爬虫模块BeautifulSoup)
前言 昨晚学的有点晚 睡得很晚了,今天早上10点多起来吃完饭看了会电视剧就瞌睡了一直睡到12.50多起来洗漱给我弟去开家长会 开到快4点多才回家.耽搁了不少学习时间,现在就把今天所学的内容总结下吧. ...
- ABBYY FineReader 14新建任务窗口给我们哪些帮助?
当您启动ABBYY FineReader时, 新任务 将打开一个窗口,在其中您可以轻松打开.扫描.创建或对比文档. 如果您没有看到此 内置任务 窗口(比如,如果您关闭了该窗口,或者您通过在 Windo ...
- Dynamics 365-表单元素取值/赋值
取值/赋值 参考: 山人丶 提示: 查找类型赋值时需指定目标实体,记录名称及id值 时间和日期类型赋值时需赋值Date类型 //获取new_name的值(单行文本) Xrm.Page.getAttri ...
- Java中的接口与抽象类的区别
由于随着jdk版本的更新,在jdk1.8时,接口也增强了,所以我们分别来说明一下. (1)jdk1.8之前 在jdk1.8之前,接口里面只能定义抽象方法和常量:而抽象类比普通类有一点不同,就是抽象类里 ...
- mybatis 动态SQL 源码解析
摘要 mybatis是个人最新喜欢的半自动ORM框架,它实现了SQL和业务逻辑的完美分割,今天我们来讨论一个问题,mybatis 是如何动态生成SQL SqlSessionManager sqlSes ...