当前位置: 首页 > news >正文

Java的动态加载及类加载器实践

在Java中,除了事先编写好代码完成执行代码,还可以从任何地方加载类。动态类加载允许程序在运行时动态编译和加载代码,打破了传统静态编译的限制。

主要的应用场景是插件系统、规则引擎、热部署等场景,本文将探讨其实现原理,并给出完整的最佳实践方案。

一、动态加载核心原理

在这里插入图片描述

二、动态加载类代码

动态编译

// 该内容是字符串形式的类代码
String sourceCode = "public class DynamicDemo { /*...*/ }";
JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();
// DynamicDemo是字符串形式类代码的类名
JavaFileObject source = new StringJavaSource("DynamicDemo", sourceCode);// 使用内存文件管理器
MemoryJavaFileManager fileManager = new MemoryJavaFileManager(compiler.getStandardFileManager(null, null, null)
);// 执行编译任务
compiler.getTask(null, fileManager, null, null, null, Collections.singletonList(source)).call();

字节码内存管理

class MemoryJavaFileObject extends SimpleJavaFileObject {private final ByteArrayOutputStream bos = new ByteArrayOutputStream();@Overridepublic OutputStream openOutputStream() {return bos; // 字节码写入内存流}public byte[] getBytes() {return bos.toByteArray();}
}

类加载

public class MemoryClassLoader extends ClassLoader {private final Map<String, byte[]> classMap;@Overrideprotected Class<?> findClass(String name) {byte[] bytes = classMap.get(name);return defineClass(name, bytes, 0, bytes.length);}
}

利用反射调用

Class<?> clazz = classLoader.loadClass("DynamicDemo");
Object instance = clazz.newInstance();
Method method = clazz.getMethod("execute");
method.invoke(instance);

三、远程加载JAR到JVM

实现步骤

在这里插入图片描述

远程JAR下载

public byte[] downloadRemoteJar(String jarUrl) throws IOException {try (InputStream in = new URL(jarUrl).openStream();ByteArrayOutputStream out = new ByteArrayOutputStream()) {byte[] buffer = new byte[4096];int bytesRead;while ((bytesRead = in.read(buffer)) != -1) {out.write(buffer, 0, bytesRead);}return out.toByteArray();}
}

内存JAR解析

public class JarInMemory {private final Map<String, byte[]> classEntries = new HashMap<>();private final Map<String, byte[]> resourceEntries = new HashMap<>();public JarInMemory(byte[] jarBytes) throws IOException {try (JarInputStream jis = new JarInputStream(new ByteArrayInputStream(jarBytes))) {JarEntry entry;while ((entry = jis.getNextJarEntry()) != null) {if (entry.isDirectory()) continue;ByteArrayOutputStream bos = new ByteArrayOutputStream();byte[] buffer = new byte[4096];int bytesRead;while ((bytesRead = jis.read(buffer)) != -1) {bos.write(buffer, 0, bytesRead);}byte[] data = bos.toByteArray();String name = entry.getName();if (name.endsWith(".class")) {String className = name.replace(".class", "").replace('/', '.');classEntries.put(className, data);} else {resourceEntries.put(name, data);}}}}
}

自定义类加载器

public class RemoteJarClassLoader extends ClassLoader {private final Map<String, byte[]> classMap;private final Map<String, byte[]> resourceMap;public RemoteJarClassLoader(ClassLoader parent, JarInMemory jar) {super(parent);this.classMap = new HashMap<>(jar.getClassEntries());this.resourceMap = new HashMap<>(jar.getResourceEntries());}@Overrideprotected Class<?> findClass(String name) throws ClassNotFoundException {byte[] classBytes = classMap.get(name);if (classBytes == null) {throw new ClassNotFoundException(name);}return defineClass(name, classBytes, 0, classBytes.length);}@Overridepublic InputStream getResourceAsStream(String name) {byte[] data = resourceMap.get(name);return data != null ? new ByteArrayInputStream(data) : null;}
}

四、更简洁的加载远程JAR

利用内置的URLClassLoader类加载器直接加载远程JAR,JVM 会按需下载 JAR 中的类文件,首次访问类时才会触发下载。加载的类遵循父委托机制,优先由父类加载器加载。

// 远程 JAR 地址
String jarUrl = "http://xxxxx.cn/abc.jar";// 创建 ClassLoader
URL[] urls = { new URL(jarUrl) };
try (URLClassLoader classLoader = new URLClassLoader(urls)) {// 加载类Class<?> clazz = classLoader.loadClass("cn.tworice.MyClass");// 实例化并调用方法Object instance = clazz.getDeclaredConstructor().newInstance();clazz.getMethod("myMethod").invoke(instance);
}

五、类加载器的注意事项

需要注意的是,不同的类加载器即使加载的是同一个类,它们之间也不能相互转换。类的唯一性由类加载器实例 + 类的全限定类名进行标识。

举个例子,我的项目中有一个业务逻辑是动态加载的类实现了BaseNode接口,但由于BaseNode和动态加载的TestNode实现类使用了不同的类加载器,则无法将动态加载的TestNode实现类转为BaseNode接口。

相关文章:

  • 《进化陷阱》--AI 生成文章 《连载 2》
  • PH热榜 | 2025-05-23
  • 板卡设计资料:基于fpga的10G以太网AD、Camera数据传输适配器
  • Pluto实验报告——基于2ASK的简易的通信系统
  • 【普及+/提高】洛谷P2613 【模板】有理数取余——快读+快速幂
  • 邻近标记技术(PL)在癌症研究中的应用
  • C语言拼接4字节数据为uint32_t
  • 数智浪潮下,解锁情绪自由密码
  • 使用DDR4控制器实现多通道数据读写(十三)
  • 六、OpenGL 2.0 通过引入可编程着色器,将渲染控制权从硬件厂商转移到开发者手中。这是如何实现的,或者说可编程着色器是如何实现的
  • 【三维重建】【3DGS系列】【深度学习】3DGS的理论基础知识之如何形成高斯椭球
  • ComfyUI Chroma解锁文生图新维度;OpenMathReasoning数学推理数据集,首个专注数学推理的高质量数据集
  • Spring Cloud实战:OpenFeign远程调用与服务治理
  • 《深度揭秘:解锁智能体大模型自我知识盲区探测》
  • 数据赋能(234)——数据管理——标准化原则
  • 第29周———Inception v3算法实战与解析
  • 探索Qwen2ForCausalLM 架构上进行微调
  • SAP Business One, Web Client: The Advantages of All Worlds
  • 【Java】Java元注解
  • YOLOv8检测头代码详解(示例展示数据变换过程)
  • 可以做网站的域名后缀/360搜索引擎入口
  • 无锡网站制作排名/简述常用的网络营销方法
  • 大数据比赛网站建设/第三方推广平台
  • 摄影网站在线建设/搜索引擎推广是什么意思
  • 需要外包团队做网站怎么提需求/seo每日工作内容
  • 做网站的公司需要哪些资质/电商网络推广是什么