读Java性能权威指南(第2版)笔记09_即时编译器下
2023-03-31 10:35:45 时间
1. CPU相关代码
1.1. Advanced Vector Extensions
1.1.1. AVX
1.1.2. 2013年,Intel为Haswell以及之后的芯片引入了AVX2
1.1.3. 2016年,Intel又引入了AVX-512指令
1.1.4. JDK 8不支持这些指令
1.1.5. JDK 11支持
1.2. -XX:UseAVX=N
1.2.1. 0
1.2.1.1. 不使用AVX指令
1.2.2. 1
1.2.2.1. 使用Intel AVX 1指令(对于Sandy Bridge和之后的处理器)
1.2.3. 2
1.2.3.1. 使用Intel AVX 2指令(对于Haswell和之后的处理器)
1.2.4. 3
1.2.4.1. 使用Intel AVX-512指令(对于Knights Landing和之后的处理器)
1.3. -XX:UseSSE=N
1.3.1. 支持Intel流SIMD扩展1到4
1.3.1.1. Streaming SIMD Extensions,SSE
1.3.2. 针对奔腾系列处理器的
2. 分层编译的权衡
2.1. 当在内存受限的环境中运行时有理由关闭它
2.2. 给定足够长的预热期,禁用分层编译时的执行情况和开启时应该是差不多的
2.3. javac编译器
2.3.1. 包含额外调试信息的-g标志不会影响性能
2.3.2. 在Java应用程序中使用final关键字并不能更快地编译代码
2.3.3. 用较新的javac版本重新编译通常不会使应用程序更快
2.3.3.1. JDK 11引入了一种新的字符串连接方式,可以比以前的版本更快,但需要重新编译代码才能用
3. GraalVM
3.1. 一个新的虚拟机
3.1.1. 可以运行许多其他语言的代码
3.2. 两个版本
3.2.1. 完全开源的社区版(Community Edition,CE)
3.2.1.1. 社区版比企业版慢
3.2.2. 商用的企业版(Enterprise Edition,EE)
3.3. 对JVM性能有两个重要贡献
3.3.1. 插件技术允许GraalVM生成完全原生的二进制文件
3.3.2. 以常规JVM的模式运行,只是它包含了一个新的C2编译器实现
3.3.2.1. 这个编译器是用Java写的
3.3.2.2. 传统的C2编译器是用C++写的
3.4. -XX:+UnlockExperimentalVMOptions
3.4.1. 默认值是false
3.5. -XX:+EnableJVMCI
3.5.1. 默认值是false
3.6. -XX:+UseJVMCICompiler
3.6.1. 默认值是false
4. 提前编译
4.1. ahead-of-time compilation,简称AOT compilation
4.2. 最初仅在JDK 9的Linux版本中可用
4.2.1. JDK 11时所有平台都可以用了
4.3. 启动速度更快
4.3.1. 目前,应用程序类数据共享给启动性能带来的提升更大,而且它已经是平台完全支持的特性
4.3.2. 提前编译针对的是像REST服务器这样启动时间相对比较长的程序
4.3.2.1. 加载共享库的时间就被较长的启动时间抵消了,提前编译就更有优势
4.4. jaotc工具
4.4.1. 生成一个共享库,其中包含你选出的编译过的类
4.4.2. 然后通过运行参数将共享库加载到JVM中
$ java -XX:+UnlockDiagnosticVMOptions -XX:+LogTouchedMethods
-XX:+PrintTouchedMethodsAtExit <other arguments>
java/net/URI.getHost:()Ljava/lang/String;
4.4.5. 为了生成methods.txt文件,需要保存这些输出内容,然后在每一行前添加compileOnly命令并删除方法参数之前的冒号
$ jaotc --compile-commands=/tmp/methods.txt
--output JavaBaseFilteredMethods.so
--compile-for-tiered
--module java.base
4.4.7. /tmp/methods.txt文件中
compileOnly java.net.URI.getHost()Ljava/lang/String;
4.5. 如果不让预编译的方法被C2编译器编译,那么服务器预热后的性能就会比它最终可能达到的性能差
4.6. -XX:+PrintAOT
4.6.1. 在预编译方法被JVM使用时会产生输出
4.6.2. 默认值是false
4.7. 对于比较大的程序有好处
4.8. 对于很小的、快速运行的程序没有帮助,甚至会阻碍它们的运行
5. GraalVM原生编译
5.1. 生成不需要JVM的可执行文件
5.1.1. 是短期程序的理想选择
5.2. 生成的二进制文件启动速度很快,特别是相较于在JVM中运行的程序
5.3. GraalVM优化代码时并没有C2编译器那么激进,所以对于运行得足够久的应用程序,传统的JVM最终会胜出
5.4. GraalVM原生二进制文件在执行期间不会使用C2编译器编译类
5.5. 原生程序的内存占用在开始时比传统JVM少得多
5.5.1. 随着程序的运行和堆的增长,这种内存优势会逐渐消失
相关文章
- 深入探讨Java中的异常与错误处理
- 研究学习Kotlin的一些方法
- 数据显示Java热度持续下落,日子屈指可数?
- 2017年5月编程语言排行榜:Java与C语言优势正开始缩小
- Java多线程之内置锁与显示锁
- Java线程池的理论与实践
- 白话阿里巴巴Java开发手册(编程规约)
- 关于Java你不知道的十件事
- Java服务化系统线上应急和技术攻关,你必须掌握的Linux命令
- Java实现高斯模糊和图像的空间卷积
- Java阻塞队列实现原理分析
- NPM使用技巧
- Node.js对Java开发者而言是什么?
- Java反射机制应用实践
- 理解RxJava中的Single和Completable
- 2017年你不能错过的Java类库
- 大规模集群下的Hadoop NameNode
- 从源码解密Spark内存管理
- 2017年3月编程语言排行榜:Swift首次进入前十
- JVM热点技术:Java类的加载机制