Java 代码编译的3种方式,其中JIT最重要!

createh55个月前 (12-22)技术教程58

通过 Javac 将程序源代码进行编译,转换成 Java 字节码,JVM 通过模板方式把字节码翻译成对应的机器指令,逐条读入,逐条解释翻译,执行速度必然比可执行的二进制字节码程序慢得多。

为了提高执行速度,引入了 JIT 技术。JIT是 JVM 的重要组成部分,JIT 通过分析程序代码,找到热点的执行代码,把部分字节码编译成机器码保存起来用于下次调用。对于较小的方法,会尝试进行内联展开。

应用程序在大部分情况下很少考虑 JIT 的优化,这是一个自动过程。不过对于性能要求极高的工具或关键服务类,还是可以考虑 JIT 对代码优化的影响,有时候性能能提高数百倍。

一、Java 的编译通常有如下方式:

1.前端编译 Javac:将 Java 源码编译成字节码。

2.提前编译 AOT:将 Java 源码编译成机器码,优点是执行速度快缺点是牺牲了平台无关性,有些优化需要在运行过程中分析确认,AOT 做不到。系统中不常用的代码也编译了。Java 9 后提供了 jaotc。

3.即时编译 JIT(Just-In-Time):字节码在执行过程中,动态编译成机器码的过程。JIT通常会分析系统的热点,对热点代码会再次尝试更加激进的优化措施从而提高 Java 系统性能.。

这里的前端编译是指将 Java 源码编译成 Java 字节码的过程,JDK 提供 javac 命令将 Java源程序编译成 java class,命令格式是 javac [options] [sourcefiles-or-classnames]。

比如,编译 Hello.java:

${java_home}/bin/javac Hello.java

前端编译 Java 源程序不一定是文件,比如,可以使用 javax.tools.JavaCompiler(JDK 6 开始支持)类,将 getSource 返回的字符串源码编译成字节码并保存到 class 文件中:

//CompileString.java
public static void main(String[] args) throws IOException {
       JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();
 
       DiagnosticCollector<JavaFileObject> diagnostics =
                     new DiagnosticCollector<>();
       StandardJavaFileManager fileManager =
compiler.getStandardFileManager(diagnostics, null, null);
              JavaStringObject stringObject =
                          new JavaStringObject("Test.java", getSource());
 
              String classes = System.getProperty("user.dir")+"/compile/javac/target/classes";
              File classesFile = new File(classes);
              fileManager.setLocation(CLASS_OUTPUT,Arrays.asList(classesFile));
 
              JavaCompiler.CompilationTask task = compiler.getTask(null,
                             fileManager, diagnostics, null, null, Arrays.asList(stringObject));
              boolean success = task.call();
              System.out.println(success?"编译成功":"编译失败");
              diagnostics.getDiagnostics().forEach(System.out::println);
}
public static String getSource() {
       return "public class Test {"
             + " }";
}

JavaCompiler 用于编译 Java 代码,javac 命令也会调用 JavaCompiler。diagnostics 用于在编译过程中保留调试、警告或者错误信息。

StandardJavaFileManager 对象用于管理源码和编译后输出的文件。本例子中设定了 CLASS_OUTPUT 目录。JavaStringObject 是自定义的一个对象,继承了 SimpleJavaFileObject,用于代表 Java 源代码,定义如下:

JavaStringObject 最重要的方法是实现了 getCharContent,提供 Java 源码。在这个例子中,Java 代码以字符的形式提供,而不是文件。

为了编译 JavaStringObject,需要创建 CompilationTask,并执行 call 方法,代码如下:

public class JavaStringObject extends SimpleJavaFileObject {
       private final String source;
       protected JavaStringObject(String name, String source) {
              super(URI.create(name), Kind.SOURCE);
              this.source = source;
       }       
       @Override
        public CharSequence getCharContent(boolean ignoreEncodingErrors)
                      throws IOException {
               return source;
         }
}

JavaStringObject 最重要的方法是实现了 getCharContent,提供 Java 源码。在这个例子中,Java 代码以字符的形式提供,而不是文件。

为了编译 JavaStringObject,需要创建 CompilationTask,并执行 call 方法,代码如下:

        JavaCompiler.CompilationTask task = compiler.getTask(null,fileManager,
diagnostics, options, null, Arrays.asList(stringObject));
        boolean success = task.call();
        System.out.println(success?"编译成功":"编译失败");
        diagnostics.getDiagnostics().forEach(System.out::println);

如果 task.call 返回 true,则表示编译成功,可以在/compile/javac/target/classes 下找到编译好的 Test.class。

代码的最后打印出编译过程中的调试、告警或者错误信息。


内容摘自《高性能Java系统权威指南》第七章

本书特点:

内容上,总结作者从事Java开发20年来在头部IT企业的高并发系统经历的真实案例,极具参考意义和可读性。

对于程序员和架构师而言,Java 系统的性能优化是一个超常规的挑战。这是因为 Java 语言和 Java 运行平台,以及 Java 生态的复杂性决定了 Java 系统的性能优化不再是简单的升级配置或者简单的 “空间换时间”的技术实现,这涉及 Java 的各种知识点。

本书从高性能、易维护、代码增强以及在微服务系统中编写Java代码的角度来描述如何实现高性能Java系统,结合真实案例,让读者能够快速上手实战。

风格上本书的风格偏实战,读者可以下载书中的示例代码并运行测试。读者可以从任意一章开始阅读,掌握性能优化知识为公司的系统所用。

本书适合:

中高级程序员和架构师;

以及有志从事基础技术研发、开源工具研发的极客阅读;

也可以作为 Java 笔试和面试的参考书。

相关文章

Java 代码执行原理 java执行过程和编译原理

专注于Java领域优质技术,欢迎关注作者 | Alan来源 | cnblogs.com/wangjiming/p/10455993.html对于任何一门语言,要想达到精通的水平,研究它的执行原理(或者...

阿里P8大牛耗费三年整理的:Java架构之完美设计实战PDF

前言做学问就像爬山,从山脚出发的人很多,能爬到山顶的却寥寥无几。有些人爬得很快,到达半山腰就盖了一所房子住在那里,对爬向山顶没有太大的兴趣。而能够登顶的人,除了要具备必要的技能外,更要有坚强的毅力,不...

不要按照 Java 的方式编写 Rust java 多线程的两种实现方式:____和____。

【CSDN 编者按】停止按照 Java 的方式编写 Rust,这是我发现编写 Rust 代码的乐趣。原文链接:https://jgayfer.com/dont-write-rust-like-java...

Java反编译工具 java反编译工具并修改代码

编译和反编译编程语言分成高级语言和低级语言。低级语言如机器语言、汇编语言。这类语言直接用计算机指令编写命令,不需要编译。这些语言机器能看到懂,但是程序员读起来很费劲。而我们平时经常用的语言C、Java...

Java 核心技术之入门指南:全面解析Java概述

大家好,这里是Java码牛!Java核心技术入门:全面解析Java概述一、引言Java ,作为一门在当今信息技术领域中被广泛应用于企业级开发的主流编程语言,其核心技术的精准掌握对于众多开发者而言,具有...