文章目录
- 概述
- Code
概述
动态编译和加载外部Java类的核心流程可以概括为以下几个步骤:
- 读取源代码: 首先,需要获取到外部的Java源代码。这通常是通过读取文件、网络资源或者数据库中的源代码字符串来实现的。
- 编译源代码: 接下来,需要使用Java编译器来编译这些源代码。这可以通过调用
javac
命令行工具或者使用Java API中的编译器API(如javax.tools.JavaCompiler
)来实现。 - 生成字节码: 编译过程会生成字节码文件(.class文件)。这些字节码文件包含了编译后的Java类的信息。
- 加载字节码: 最后,需要将这些字节码文件加载到Java虚拟机(JVM)中。这可以通过创建一个
ClassLoader
子类并重写其loadClass
方法来实现。在这个方法中,你可以从文件系统、网络或其他来源读取字节码,并使用defineClass
方法将其定义为一个Class
对象。 - 创建实例和调用方法: 一旦类被加载,就可以使用
newInstance
方法来创建类的实例,并调用其方法。
Code
package com.artisan.jsr269;import javax.tools.JavaCompiler;
import javax.tools.StandardJavaFileManager;
import javax.tools.ToolProvider;
import java.io.File;
import java.io.FileWriter;
import java.lang.reflect.Method;
import java.net.URL;
import java.net.URLClassLoader;/*** @author 小工匠* @version 1.0* @mark: show me the code , change the world*/
public class DynamicCompiler {public static void main(String[] args) throws Exception {//创建源文件String currentDir = System.getProperty("user.dir") + "/boot-beanUtils" ;// 定义一个简单的Java类,包含一个方法,该方法打印出“Hello Artisan”String src = "package com.artisan.jsr269 ;"+ "public class ArtisanComplier {"+ " public void methodA() {"+ " System.out.println(\"Hello Artisan\");"+ "}}";// 源文件路径和名称String filename = currentDir + "/src/main/java/com/artisan/jsr269/ArtisanComplier.java";File file = new File(filename);// 确保源文件所在的目录存在File fileParent = file.getParentFile();if (!fileParent.exists()) {fileParent.mkdir();}// 确保源文件存在if (!file.exists()) {file.createNewFile();}// 将源代码写入文件FileWriter fw = new FileWriter(file);fw.write(src);fw.flush();fw.close();// 使用JavaCompiler 编译java文件// 获取系统Java编译器JavaCompiler jc = ToolProvider.getSystemJavaCompiler();// 获取标准文件管理器StandardJavaFileManager fileManager = jc.getStandardFileManager(null, null, null);// 获取要编译的文件对象Iterable fileObjects = fileManager.getJavaFileObjects(filename);// 创建编译任务JavaCompiler.CompilationTask cTask = jc.getTask(null, fileManager, null, null, null, fileObjects);// 执行编译任务cTask.call();// 关闭文件管理器fileManager.close();// 使用URLClassLoader加载class到内存URL[] urls = new URL[]{new URL("file:/" + currentDir + "/src/main/java/com/artisan/jsr269/ArtisanComplier.java")};URLClassLoader cLoader = new URLClassLoader(urls);// 加载类Class c = cLoader.loadClass("com.artisan.jsr269.ArtisanComplier");// 关闭类加载器cLoader.close();// TODO 在这之前要确保编译任务完成,否则这里通过反射实例化会报错// 利用class创建实例,反射执行方法Object obj = c.newInstance();// 获取类中的方法Method method = c.getMethod("methodA");// 执行方法method.invoke(obj);}
}
运行结果