动态编译 Java 源码为 Class
一.双亲委派
Java 开发,如果要运行程序,首先就是编译源码,其次是程序包启动加载类,最后程序运行。
众所周知 Java 是基于 Jvm 虚拟机运行的,那么程序启动时怎么区分开发人员定义的类
和 Jvm 提供的类呢;如果不区分,万一开发人员定义了一个 Stirng 类,很有可能导致 Jvm 瘫痪...
为了避免上述情况, 类加载时采用了双亲委派模型,即向上委托,向下查找:
某些官方类由顶级类加载器加载,这样开发人员引入官方类时,向上委托加载;
如果引入自定义类,上层加载器查不到,则向下查找,最终有应用层类加载器,加载自定义类
同时类加载时,已加载过的类会直接返回,不会重复加载
所以,如果要动态编译源码,增加新的类,新的处理逻辑比较容易;想要修改已有的类,并让程序引用到新的类,比较困难,因为已加载的类如果不卸载,是不会重新加载同名类的;同时,双亲委派模型是可以打破的,比较典型的实现有 Tomcat
关于 Jvm 类加载,双亲委派,和打破双亲委派的更多细节,大家可以自行学习
二.SpringBoot 应用动态编译源码
1.Pom 依赖
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>org.example</groupId>
<artifactId>DynamicDemo</artifactId>
<version>1.0-SNAPSHOT</version>
<properties>
<maven.compiler.source>19</maven.compiler.source>
<maven.compiler.target>19</maven.compiler.target>
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
</properties>
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
<version>2.7.2</version>
</dependency>
</dependencies>
</project>
2.SpringBoot IOC 容器
SpringBoot 最重要的实现就是 IOC 和 AOP ,IOC 使我们可以通过注解方便的引用单例类对象
如果要基于 SpringBoot 应用实现源码动态编译,我们可以定义一个 SpringBoot 的 Bean 工具类,用于移除和注册 Bean 对象,同时如果可能设计修改的类,需要用抽象类或接口定义,此时修改实现类,则相当于修改方法;这样,基于上一篇的自定义类加载器加载新的类,可以避过同名类无法重复加载的问题
package org.example.demo.util;
import org.springframework.beans.BeansException;
import org.springframework.beans.factory.config.BeanDefinition;
import org.springframework.beans.factory.support.BeanDefinitionBuilder;
import org.springframework.beans.factory.support.DefaultListableBeanFactory;
import org.springframework.context.ApplicationContext;
import org.springframework