java动态编译

编译,一般来说就是将源代码转换成机器码的过程,比如在C语言中中,将C语言源代码编译成a.out,,但是在Java中的理解可能有点不同,编译指的是将java 源代码转换成class字节码的过程,而不是真正的机器码,这是因为中间隔着一个JVM。虽然对于编译的理解不同,但是编译的过程基本上都是相同的。但是我们熟悉的编译大都是点击一下Eclipse或者Intellij Idea的Run或者Build按钮,但是在点击后究竟发生什么?其实我没仔细了解过,只是知道这个程序运行起来了,但是如果你使用过javac命令去编译代码时,可能了解的就更深一些,据说印度的Java程序员最开始编程的时候使用的都是文本编辑器而不是IDE,这样更能接触底层的过程。
除了使用javac命令编译Java程序,从Java 1.6开始,我们也可以在程序运行时根据程序实际运行来构建一些类并进行编译,这需要JDK提供给我们一些可供调用的接口来完成编译工作。
一、编译源码需要啥?
那么问题来了,如果要了解运行时编译的过程和对应的接口,首先要明白的就是编译这个过程都会涉及哪些工具和要解决的问题?从我们熟悉的构建过程开始:

编译工具(编译器):显然没有这个东西我们啥也干不了;
要编译的源代码文件:没有这个东西,到底编啥呢?
源代码、字节码文件的管理:其实这里靠的是文件系统的支持,包括文件的创建和管理;
编译过程中的选项:要编译的代码版本、目标,源代码位置,classpath和编码等等,见相关文章;
编译中编译器输出的诊断信息:告诉你编译成功还是失败,会有什么隐患提出警告信息;
按照这些信息,JDK也提供了可编程的接口对象上述信息,这些API全部放在javax.tools包下,对应上面的信息如下:
编译器:涉及到的接口和类如下:

JavaCompiler
JavaCompiler.CompilationTask
ToolProvider
在上面的接口和类中,ToolProvider类似是一个工具箱,它可以提供JavaCompiler类的实例并返回,然后该实例可以获取JavaCompiler.CompilationTask实例,然后由JavaCompiler.CompilationTask实例来执行对应的编译任务,其实这个执行过程是一个并发的过程。

源代码文件:涉及到接口和类如下:

FileObject
ForwardingFileObject
JavaFileObject
JavaFileObject.Kind
ForwardingJavaFileObject
SimpleJavaFileObject
上述后面的4个接口和类都是FileObject子接口或者实现类,FIleObject接口代表了对文件的一种抽象,可以包括普通的文件,也可以包括数据库中的数据库等,其中规定了一些操作,包括读写操作,读取信息,删除文件等操作。我们要用的其实是JavaFileObject接口,其中还增加了一些操作Java源文件和字节码文件特有的API,而SimpleJavaFileObject是JavaFileObject接口的实现类,但是其中你可以发现很多的接口其实就是直接返回一个值,或者抛出一个异常,并且该类的构造器由protected修饰的,所以要实现复杂的功能,需要我们必须扩展这个类。ForwardingFileObject、ForwardingJavaFileObject类似,其中都是包含了对应的FileObject和JavaFileObject,并将方法的执行委托给这些对象,它的目的其实就是为了提高扩展性。

文件的创建和管理:涉及接口和类如下:

JavaFileManager
JavaFileManager.Location
StandardJavaFileManager
ForwardingJavaFileManager
StandardLocation

JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();
DiagnosticCollector<JavaFileObject> collector = new DiagnosticCollector<>();
// 该JavaFileManager实例是com.sun.tools.javac.file.JavacFileManager
JavaFileManager manager= compiler.getStandardFileManager(collector, null, null);

JavaFileManager用来创建JavaFileObject,包括从特定位置输出和输入一个JavaFileObject,ForwardingJavaFileManager也是出于委托的目的。而StandardJavaFileManager是JavaFileManager直接实现类,JavaFileManager.Location和StandardLocation描述的是JavaFileObject对象的位置,由JavaFileManager使用来决定在哪创建或者搜索文件。由于在javax.tools包下没有JavaFileManager对象的实现类,如果我们想要使用,可以自己实现该接口,也可以通过JavaCompiler类中的getStandardFileManager完成,如下:

编译选项的管理:

OptionChecker
这个接口基本上没有用过。

诊断信息的收集:涉及接口和类如下:

Diagnostic
DiagnosticListener
Diagnostic.Kind
DiagnosticCollector
Diagnostic会输出编译过程中产生的问题,包括问题的信息和出现问题的定位信息,问题的类别则在Diagnostic.Kind中定义。DiagnosticListener则是从编译器中获取诊断信息,当出现诊断信息时则会调用其中的report方法,DiagnosticCollector则是进一步实现了DiagnosticListener,并将诊断信息收集到一个list中以便处理。

在Java源码运行时编译的时候还会遇到一个与普通编译不同的问题,就是类加载器的问题,由于这个问题过大,而且比较核心,将会专门写一篇文章介绍。
二、如何在运行时编译源代码?
好了说了这么多了,其实都是为了下面的实例作为铺垫,我们还是从上述的几个组件来说明。

1、准备编译器对象
这里只有一种方法,如下:

JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();
// ......
// 在其他实例都已经准备完毕后, 构建编译任务, 其他实例的构建见如下
Boolean result = compiler.getTask(null, manager, collector, options,null,Arrays.asList(javaFileObject));

2、诊断信息的收集

// 初始化诊断收集器
DiagnosticCollector<JavaFileObject> collector = new DiagnosticCollector<>();
// ......
// 编译完成之后,获取编译过程中的诊断信息
collector.getDiagnostics().forEach(item -> System.out.println(item.toString()))

在这个过程中可以通过Diagnostic实例获取编译过程中出错的行号、位置以及错误原因等信息。
3、源代码文件对象的构建
由于JDK提供的FileObject、ForwardingFileObject、JavaFileObject、ForwardingJavaFileObject、SimpleJavaFileObject都无法直接使用,所以我们需要根据需求自定义,此时我们要明白SimpleJavaFileObject类中的哪些方法是必须要覆盖的,可以看如下过程:

下面是调用compiler中的getTask方法时的调用栈,可以看出从main()方法中开始调用getTask方法开始,直到编译工作开始进行,首先读取源代码,调用com.sun.tools.javac.main包中的readSource()方法,源代码如下:

public CharSequence readSource(JavaFileObject filename) {try {inputFiles.add(filename);return filename.getCharContent(false);} catch (IOException e) {log.error("error.reading.file", filename, JavacFileManager.getMessage(e));return null;}
}

其中调用ClientCodeWrapper$WrappedFileObject对象中的filename.getCharContent(false)方法来读取要编译的源码,源代码如下:

public CharSequence getCharContent(boolean ignoreEncodingErrors) throws IOException {try {return clientFileObject.getCharContent(ignoreEncodingErrors);} catch (ClientCodeException e) {throw e;} catch (RuntimeException e) {throw new ClientCodeException(e);} catch (Error e) {throw new ClientCodeException(e);}
}

而其中的clientFileObject.getCharContent(ignoreEncodingErrors),其实就是调用我们实现的自定义的JavaFIleObject对象,因此源代码文本是必须的,因此getCharContent方法是必须实现的,另外在编译器编译完成之后要将编译完成的字节码输出,如下图:

这时调用writeClass()输出字节码,通过打开一个输出流OutputStream来完成该过程,因此openOutputStream()这个方法也是必须实现的。因此该类的实现如下:

public static class MyJavaFileObject extends SimpleJavaFileObject {private String source;private ByteArrayOutputStream outPutStream;// 该构造器用来输入源代码public MyJavaFileObject(String name, String source) {// 1、先初始化父类,由于该URI是通过类名来完成的,必须以.java结尾。// 2、如果是一个真实的路径,比如是file:///test/demo/Hello.java则不需要特别加.java// 3、这里加的String:///并不是一个真正的URL的schema, 只是为了区分来源super(URI.create("String:///" + name + Kind.SOURCE.extension), Kind.SOURCE);this.source = source;}// 该构造器用来输出字节码public MyJavaFileObject(String name, Kind kind){super(URI.create("String:///" + name + kind.extension), kind);source = null;}@Overridepublic CharSequence getCharContent(boolean ignoreEncodingErrors){if(source == null){throw new IllegalArgumentException("source == null");}return source;}@Overridepublic OutputStream openOutputStream() throws IOException {outPutStream = new ByteArrayOutputStream();return outPutStream;}// 获取编译成功的字节码byte[]public byte[] getCompiledBytes(){return outPutStream.toByteArray();}
}

4、文件管理器对象的构建
文件管理对象显然也是不能直接使用JDK提供的接口,因为只有ForwardingJavaFileManager是一个类,其他的都是接口,而且在ForwardingJavaFileManager中构造器又是protected,所以如果想定制化使用的话,需要实现接口或者继承类,如果只是简单使用,可以如下:

JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();
DiagnosticCollector<JavaFileObject> collector = new DiagnosticCollector<>();
// 该JavaFileManager实例是com.sun.tools.javac.file.JavacFileManager
JavaFileManager manager= compiler.getStandardFileManager(collector, null, null);

但是compiler.getStandardFileManager()返回的是com.sun.tools.javac.file.JavacFileManager实例,这个不是公开的类,所以我们无法直接使用,只能通过这种调用返回实例。

但是我们课也可以构造自己的FileManager,为了更好的构建,需要理解JavaFileManager在内存中编译时的使用过程,如下:

在编译过程中,首先是编译器会遍历JavaFileManager对象,获取指定位置的所有符合要求的JavaFileObject对象,甚至可以递归遍历,这时调用的是list()方法,该方法会扫面所有涉及的到的包,包括一个类和它实现的接口和继承的类:

之后根据获取到的JavaFileObject对象,获取它的二进制表示的名称,通过调用inferBinaryName()方法;

之后是输出编译类,而类的表示为JavaFileObject对象,注意此时的JavaFileObject.Kind为CLASS,调用的方法是getJavaFileForOutput(),注意该方法的调用是在JavaFileObject中openOutputStream()方法之前,如下图:

既然了解了上述的流程,我们自定义的文件管理器如下:

private static Map<String, JavaFileObject> fileObjects = new ConcurrentHashMap<>();
// 这里继承类,不实现接口是为了避免实现过多的方法
public static class MyJavaFileManager extends ForwardingJavaFileManager<JavaFileManager> {protected MyJavaFileManager(JavaFileManager fileManager) {super(fileManager);}@Overridepublic JavaFileObject getJavaFileForInput(Location location, String className, JavaFileObject.Kind kind) throws IOException {JavaFileObject javaFileObject = fileObjects.get(className);if(javaFileObject == null){super.getJavaFileForInput(location, className, kind);}return javaFileObject;}@Overridepublic JavaFileObject getJavaFileForOutput(Location location, String qualifiedClassName, JavaFileObject.Kind kind, FileObject sibling) throws IOException {JavaFileObject javaFileObject = new MyJavaFileObject(qualifiedClassName, kind);fileObjects.put(qualifiedClassName, javaFileObject);return javaFileObject;}
}

5、编译选项的选择
在使用javac命令的时候,可以添加很多的选项,在实现API完成编译的时候也可以提供参数,比如编译目标,输出路径以及类路径等等,如下:

List<String> options = new ArrayList<>();
options.add("-target");
options.add("1.8");
options.add("-d");
options.add("/");
// 省略......
compiler.getTask(null, javaFileManager, collector, options, null, Arrays.asList(javaFileObject));

6、其他问题
想将编译完成的字节码输出为文件,也不需要上面自定义JavaFileManager,直接使用JavaCompiler提供的即可,而且在自定义的JavaFileObject中也不需要实现OpenOutStream这种方法,代替要提供options.add(“-d”),options.add(“/”)等编译选项;如果不输出为文件按照上述的例子即可;
StandardLocation中的元素可以代替真实的路径位置,但是不会输出为文件,可以为一个内存中的文件;
在编译完成之后要将字节码文件加载进来,因此就要涉及到类加载机制,由于这也是一个很大的话题,所以后面会专门总结一篇,但是在这里还是要说明一下,由于上面编译时没有额外的依赖包,所以不用考虑加载依赖文件的问题,但是当如果有这样的需求时,我们可以利用类加载的委托机制,将依赖文件的加载全部交给父加载器去做即可。
完整的代码如下:

package com.wdx.compiler;import org.slf4j.Logger;
import org.slf4j.LoggerFactory;import javax.tools.Diagnostic;
import javax.tools.DiagnosticCollector;
import javax.tools.FileObject;
import javax.tools.ForwardingJavaFileManager;
import javax.tools.JavaCompiler;
import javax.tools.JavaFileManager;
import javax.tools.JavaFileObject;
import javax.tools.SimpleJavaFileObject;
import javax.tools.ToolProvider;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.net.URI;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;public class CubeJavaCompiler{private static final Logger logger = LoggerFactory.getLogger(CubeJavaCompiler.class);private static final JavaCompiler _compiler = ToolProvider.getSystemJavaCompiler();private static final DiagnosticCollector<JavaFileObject>  collector = new DiagnosticCollector<>();private static final CubeJavaFileManager manager = new CubeJavaFileManager(_compiler.getStandardFileManager(collector, null, null));private static final Map<String, JavaFileObject> fileObjectMap = new ConcurrentHashMap<>();private static List<String> options = new ArrayList<>();static {options.add("-Xlint:unchecked");options.add("-target");options.add("1.8");}public static Class<?> compile(String code, String className) throws ClassNotFoundException{String qualified = className.substring(className.lastIndexOf('.') + 1, className.length());CubeJavaObject cubeJavaObject = new CubeJavaObject(qualified, code);JavaCompiler.CompilationTask task = _compiler.getTask(null, manager, collector, options, null, Arrays.asList(cubeJavaObject));task.call();//输出诊断信息for (Diagnostic<? extends JavaFileObject> diagnostic : collector.getDiagnostics()) {try {logger.error("编译错误:{}", diagnostic.toString());} catch (Exception e) {logger.error("输出内容错误", e);}}return cubeJavaClassLoader.loadClass(className);}private static ClassLoader cubeJavaClassLoader = new ClassLoader() {@Overrideprotected Class<?> findClass(String name) throws ClassNotFoundException {JavaFileObject fileObject = fileObjectMap.get(name);if(fileObject != null){byte[] bytes = ((CubeJavaObject)fileObject).getCompiledBytes();return defineClass(name, bytes, 0, bytes.length);}try{return ClassLoader.getSystemClassLoader().loadClass(name);} catch (Exception e){logger.error("加载类失败,{}", name, e);return super.findClass(name);}}};private static class CubeJavaObject extends SimpleJavaFileObject{private String code;private ByteArrayOutputStream outPutStream;public CubeJavaObject(String qualified, String code) {super(URI.create("String:///" + qualified + Kind.SOURCE.extension), Kind.SOURCE);this.code = code;}public CubeJavaObject(String qualified, Kind kind) {super(URI.create("String:///" + qualified + kind.extension), kind);}@Overridepublic CharSequence getCharContent(boolean ignoreEncodingErrors) throws IOException {if(code == null){throw new IllegalArgumentException("code required");}return code;}@Overridepublic OutputStream openOutputStream() throws IOException {outPutStream = new ByteArrayOutputStream();return outPutStream;}public byte[] getCompiledBytes(){return outPutStream.toByteArray();}}private static class CubeJavaFileManager extends ForwardingJavaFileManager<JavaFileManager> {public CubeJavaFileManager(JavaFileManager fileManager) {super(fileManager);}@Overridepublic JavaFileObject getJavaFileForInput(Location location, String className, JavaFileObject.Kind kind) throws IOException {JavaFileObject javaFileObject = fileObjectMap.get(className);if(javaFileObject == null){super.getJavaFileForInput(location, className, kind);}return javaFileObject;}@Overridepublic JavaFileObject getJavaFileForOutput(Location location, String className, JavaFileObject.Kind kind, FileObject sibling) throws IOException {JavaFileObject javaFileObject = new CubeJavaObject(className, kind);fileObjectMap.put(className, javaFileObject);return javaFileObject;}}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/250058.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

[c++] - 简单的冒泡

#include <iostream> using namespace std;int main() {// 利用冒泡排序实现升序序列int arr[9] {4, 2, 8, 0, 5, 7, 1, 3, 9};cout << "排序前: " << endl;for (int i 0; i < 9; i){cout << arr[i] << " ";}cout <…

Python爬虫之解析网页

常用的类库为lxml, BeautifulSoup, re(正则) 以获取豆瓣电影正在热映的电影名为例,urlhttps://movie.douban.com/cinema/nowplaying/beijing/ 网页分析 部分网页源码 <ul class"lists"><liid"3878007"class"list-item"data-title"…

腾讯企业邮箱报错 smtp.exmail.qq.comport 465, isSSL false

一、报错 "smtp.exmail.qq.com" port 465, isSSL false 通过网上搜索查询一些资料&#xff0c;推测是邮箱的配置出问题了。 二、修改邮箱配置 1 // 创建属性2 Properties props new Properties();3 props.setProperty("mail.transport.protocol", "s…

spring与JDK版本对应关系

搭建spring框架得时候要考虑jdk的版本&#xff0c;提供一下参考 JDK 8 中可以使用 Spring Framework 5.x JDK 7 中可以使用 Spring Framework 4.x JDK 6 中可以使用 Spring Framework 4.x JDK 5 中可以使用 Spring Framework 3.x

Markdown预览功能不可用解决方案

初学者在使用Markdown时也许会遇到这个问题 原因是电脑缺少一个组件&#xff0c;解决方案很简单&#xff0c;安装上就好了&#xff0c;以下是链接 http://markdownpad.com/download/awesomium_v1.6.6_sdk_win.exe转载于:https://www.cnblogs.com/j9oker/p/10092829.html

Linux 中yum的配置

1.进入yum的路径 cd /etc/yum.repos.d 2.将原始的repo文件移入一个新建的backup文件下做备份 mv CentOS* backup 3.在/etc/yum.repos.d下新建一个自己的文件(这里的文件必须以repo结尾); vi zhi.repo 其中&#xff0c;第一行必须是[文件名]的格式  是一个标记 name*** 这是一…

[生态建设] - js判断小技巧

0、参考 说明: 从几个易得的点出发,逐步向外扩展延申,保证代码的可靠性 1、判断是否为某个类型 // 判断是否为 null const isNull o > {return o null; };// 判断是否为 undefined const isUndefined o > {return o undefined; };// 判断是否为 null or undefined…

Spring中Bean的概念

一、Bean的定义 <beans…/>元素是Spring配置文件的根元素&#xff0c;<beans…/>元素可以包含多个<bean…/>子元素&#xff0c;每个<bean…/>元素可以定义一个Bean实例&#xff0c;每一个Bean对应Spring容器里的一个Java实例定义Bean时通常需要指定两…

[TJOI2010]阅读理解

题目描述 英语老师留了N篇阅读理解作业&#xff0c;但是每篇英文短文都有很多生词需要查字典&#xff0c;为了节约时间&#xff0c;现在要做个统计&#xff0c;算一算某些生词都在哪几篇短文中出现过。 输入输出格式 输入格式&#xff1a; 第一行为整数N&#xff0c;表示短文篇…

ccentos 7下安装php5.6并使用nginx + php-fpm部署多个不同端口网站

作为一个的勤杂工&#xff0c;近期因公司内部信息化的需求&#xff0c;给新进员工提供基础的知识培训和介绍&#xff0c;也为了给公司内部建立一个沟通交流的平台&#xff0c;百度找了开源的百科系统HDwiki和开源的问答系统Tipask问答系统&#xff0c;蛋痛的这两套系统均是phpm…

Zookeeper基础使用机制原理

Znode&#xff1a; 1、Znode既是路径(目录)也是信息(文件) 2、Znode有两种分类&#xff1a;一分为临时节点(会话生命周期)和永久节点&#xff1b;二分为普通节点和顺序节点 Watch&#xff1a; 1、监听与通知机制&#xff0c;可以在节点上监听其本身(增、删、改)或其子节点(增、…

JS ajax请求参数格式( formData 、serialize)

1 $("#importBtn").click(function(){2 if($("#conId").val() ""){3 alert("请填写Id");4 return;5 }6 if($("#fromWhere").val() "…

【小工具分享】 - vscode注释自动生成

参考 关闭文件头部注释 点击设置 输入fileheader搜索 关闭头部注释 "fileheader.customMade" : {"autoAdd": false }

Spring的bean实例化过程

以XmlBeanFactory为例&#xff0c;最简单的取bean方式是&#xff1a; BeanFactory factory new XmlBeanFactory(new FileSystemResource("D:\\workspace\\JavaApplication2\\src\\javaapplication2\\spring\\beans.xml")); Car obj (Car)factory.getBean("c…

最全整理浏览器兼容性问题与解决方案(转)

所谓的浏览器兼容性问题&#xff0c;是指因为不同的浏览器对同一段代码有不同的解析&#xff0c;造成页面显示效果不统一的情况。在大多数情况下&#xff0c;我们的需求是&#xff0c;无论用户用什么浏览器来查看我们的网站或者登陆我们的系统&#xff0c;都应该是统一的显示效…

【算法】 - 滑动窗口

1. 题目链接 2. 分析 最多可以将K个值从0变成1,因此滑动窗口的限制条件: 0的数量(zeros)小于K,算法过程如下 有一个滑动窗口(slipper),每次都会从A中读入一个数当读入的数为0时,zeros当zeros的数量大于K时,会取出slipper首部的元素,当取值为0时zeros-- 总体代码如下: var lo…

Springboot整合thymeleaf模板

Thymeleaf是个XML/XHTML/HTML5模板引擎&#xff0c;可以用于Web与非Web应用。 Thymeleaf的主要目标在于提供一种可被浏览器正确显示的、格式良好的模板创建方式&#xff0c;因此也可以用作静态建模。你可以使用它创建经过验证的XML与HTML模板。相对于编写逻辑或代码&#xff0…

Java代码输出到txt文件(申请专利贴源码的必备利器)

最近公司在申请专利&#xff0c;编写不少文档&#xff0c;项目的代码量实在是过于庞大。如果一个一个的复制粘贴虽然能够完成&#xff0c;但是对于程序员而言实在没有这个必要。shell或者python就能解决这个问题。由于我个人对于shell和python不是非常熟练的情况下&#xff0c;…

【算法】 - 动态规划 + 位运算

题目描述 思路1: 写一个返回2进制中1数量的函数countOne遍历0到num,对每一个数使用countOne,并将结果保存到res中返回 var countBits function (num) {let res new Array(num 1).fill(0);for (let i 0; i < num; i) {res[i] countOne(i.toString(2));}return res; };…