java~通过ClassLoader动态加载类,实现简单的热部署
热部署一般是在开发中使用,在spring里一般使用devtools,springloaded等工具来实现,一般在调试项目时不需要再发布,而直接加载变化的类对象即可;而有时有生产环境也需要考虑对一些jar进行热部署。
- 说明:本方法是加载外部包里的类,然后通过反射实现,如果你引用了这个包到项目里,那本方法将失效
一个类加载器只能加载一个同名类,在Java默认的类加载器层面作了判断,如果已经有了该类,则不再重复加载,如果强行绕过判断并使用自定义类加载器重复加载,JVM 将会抛出 LinkageError:attempted duplicate class definition for name。
注意:不同的类加载器是可以加载同名的类的,加载完成之后,这两个类虽然同名,但不是同一个 Class 对象,使用自定义的类加载器,加载一个类,当需要进行替换类的时候,我们就丢弃之前的类加载器和类,使用新的类加载器去加载新的 Class 文件,然后运行新对象的方法。
热部署的过程
接口调用
通过一个对外的接口,进行数据的返回,主要调用了HelloImpl类里的password方法
/**
* 自定义类加载器.
*/
@Component
public class RsaClassLoader {
@SneakyThrows
public Class findClassLoader(String packageUrl, String name) throws ClassNotFoundException {
URL url = new URL(packageUrl);
ClassLoader loader = new URLClassLoader(new URL[]{url}) {
@Override
public Class<?> loadClass(String name) throws ClassNotFoundException {
try {
String fileName = name.substring(name.lastIndexOf(".") + 1) + ".class";
InputStream is = getClass().getResourceAsStream(fileName);
if (is == null) {
return super.loadClass(name);
}
byte[] b = new byte[is.available()];
is.read(b);
return defineClass(name, b, 0, b.length);
} catch (IOException e) {
e.printStackTrace();
throw new ClassNotFoundException(name);
}
}
};
return loader.loadClass(name);
}
}
在接口中调这个公用类
Class clazz = rsaClassLoader.findClassLoader("file:///D:\\a-start-hot-dependency-1.0.0.jar","com.lind.hot.HelloImpl");
Object account = clazz.newInstance();
Object ret = account.getClass().getMethod("password", new Class[]{}).invoke(account);
return ResponseEntity.ok(ret);
在程序运行时,你可以动态修改a-start-hot-dependency-1.0.0.jar这个包的内容,来达到热部署的效果!