手写一个动态编译Java源码并加载Class
2022/3/21 11:27:42
本文主要是介绍手写一个动态编译Java源码并加载Class,对大家解决编程问题具有一定的参考价值,需要的程序猿们随着小编来一起学习吧!
import javax.tools.SimpleJavaFileObject; import java.net.URI; /** * 将代码保存在字符串构建器中的 Java 源代码。 * * @description: * @author: mabh * @create: 2022/3/19 22:33 **/ public class StringBuilderJavaSource extends SimpleJavaFileObject { private StringBuilder code; public StringBuilderJavaSource(String name) { super(URI.create("string:///" + name.replace(".", "/") + Kind.SOURCE.extension), Kind.SOURCE); code = new StringBuilder(); } @Override public CharSequence getCharContent(boolean ignoreEncodingErrors) { return code; } public void append(String str) { code.append(str); code.append("\n"); } }
import java.util.Map; public class MapClassLoader extends ClassLoader { private final Map<String, byte[]> classes; public MapClassLoader(Map<String, byte[]> classes) { this.classes = classes; } @Override protected Class<?> findClass(String name) throws ClassNotFoundException { byte[] classBytes = classes.get(name); if (classBytes == null) { throw new ClassNotFoundException(name); } Class<?> cl = defineClass(name, classBytes, 0, classBytes.length); if (cl == null) { throw new ClassNotFoundException(name); } return cl; } }
import java.util.Map; public class MapClassLoader extends ClassLoader { private final Map<String, byte[]> classes; public MapClassLoader(Map<String, byte[]> classes) { this.classes = classes; } @Override protected Class<?> findClass(String name) throws ClassNotFoundException { byte[] classBytes = classes.get(name); if (classBytes == null) { throw new ClassNotFoundException(name); } Class<?> cl = defineClass(name, classBytes, 0, classBytes.length); if (cl == null) { throw new ClassNotFoundException(name); } return cl; } }
需要一个测试类,我们新建一个接口
/** * @description: * @author: mabh * @create: 2022/3/19 23:15 **/ public interface IABC { void main2(); }
重头戏
import javax.tools.*; import java.io.IOException; import java.util.*; /** * @description: * @author: mabh * @create: 2022/3/19 21:49 **/ public class Test222{ public static void main(String[] args) throws Exception { JavaCompiler compiler = ToolProvider.getSystemJavaCompiler(); List<ByteArrayJavaClass> classFileObjects = new ArrayList<>(); DiagnosticCollector<JavaFileObject> diagnostic = new DiagnosticCollector<>(); JavaFileManager fileManager = compiler.getStandardFileManager(diagnostic, null, null); fileManager = new ForwardingJavaFileManager<JavaFileManager>(fileManager) { @Override public JavaFileObject getJavaFileForOutput(Location location, String className, JavaFileObject.Kind kind, FileObject sibling) throws IOException { ByteArrayJavaClass arrayJavaClass = new ByteArrayJavaClass(className); classFileObjects.add(arrayJavaClass); return arrayJavaClass; } }; StringBuilderJavaSource javaSource = new StringBuilderJavaSource("ABC"); javaSource.append("package com.epoch.planning;"); javaSource.append(" public class ABC implements IABC {"); javaSource.append(" public void main2() {"); javaSource.append(" System.out.print(\"我是使用java动态编译的类\");"); javaSource.append(" }"); javaSource.append(" }"); JavaCompiler.CompilationTask compilerTask = compiler.getTask(null, fileManager, diagnostic, null, null, Arrays.asList(javaSource)); Boolean result = compilerTask.call(); for (Diagnostic<? extends JavaFileObject> diagnosticDiagnostic : diagnostic.getDiagnostics()) { System.out.println(diagnosticDiagnostic.getKind() + ": " + diagnosticDiagnostic.getMessage(null)); } fileManager.close(); if (!result) { List<Diagnostic<? extends JavaFileObject>> diagnostics = diagnostic.getDiagnostics(); for (Diagnostic<? extends JavaFileObject> diagnostic1 : diagnostics) { System.out.println(diagnostic1.toString()); break; } System.out.println("compile failed!"); } Map<String, byte[]> byteCodeMap = new HashMap<>(); for (ByteArrayJavaClass classFileObject : classFileObjects) { byteCodeMap.put(classFileObject.getName().substring(1), classFileObject.getBytes()); } MapClassLoader loader = new MapClassLoader(byteCodeMap); IABC abc = (IABC) loader.findClass("com.epoch.planning.ABC").newInstance(); abc.main2(); } }
失败了会自动打印日志
修改正确即可正确输出
这篇关于手写一个动态编译Java源码并加载Class的文章就介绍到这儿,希望我们推荐的文章对大家有所帮助,也希望大家多多支持为之网!
- 2024-11-27消息中间件底层原理资料详解
- 2024-11-27RocketMQ底层原理资料详解:新手入门教程
- 2024-11-27MQ底层原理资料详解:新手入门教程
- 2024-11-27MQ项目开发资料入门教程
- 2024-11-27RocketMQ源码资料详解:新手入门教程
- 2024-11-27本地多文件上传简易教程
- 2024-11-26消息中间件源码剖析教程
- 2024-11-26JAVA语音识别项目资料的收集与应用
- 2024-11-26Java语音识别项目资料:入门级教程与实战指南
- 2024-11-26SpringAI:Java 开发的智能新利器