熱部署一般是在開發中使用,在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這個包的內容,來達到熱部署的效果!