java 动态加载修改类_java~通过ClassLoader动态加载类,实现简单的热部署

热部署一般是在开发中使用,在spring里一般使用devtools,springloaded等工具来实现,一般在调试项目时不需要再发布,而直接加载变化的类对象即可;而有时有生产环境也需要考虑对一些jar进行热部署。

说明:本方法是加载外部包里的类,然后通过反射实现,如果你引用了这个包到项目里,那本方法将失效

一个类加载器只能加载一个同名类,在Java默认的类加载器层面作了判断,如果已经有了该类,则不再重复加载,如果强行绕过判断并使用自定义类加载器重复加载,JVM 将会抛出 LinkageError:attempted duplicate class definition for name。

注意:不同的类加载器是可以加载同名的类的,加载完成之后,这两个类虽然同名,但不是同一个 Class 对象,使用自定义的类加载器,加载一个类,当需要进行替换类的时候,我们就丢弃之前的类加载器和类,使用新的类加载器去加载新的 Class 文件,然后运行新对象的方法。

热部署的过程

8e82cd5aef00733b65b31b54b1999b56.png

接口调用

通过一个对外的接口,进行数据的返回,主要调用了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这个包的内容,来达到热部署的效果!

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值