小引子
最近做了一個根據同一模塊的不同jar版本做同時測試的工具,感覺挺有意思,特此記錄。
類加載器(ClassLoader)是啥?
把類加載階段中的“通過一個類的全限定名(博主注:絕對路徑)來獲取描述此類的二進制字節流”這個動作放在Java虛擬機外部去實現,以便讓應用程序自己決定如何去獲取所需要的類。實現這個動作的代碼模塊成為”類加載器“。摘自周志明的《深入理解Java虛擬機》
ClassLoader的用途
- 功能測試
每個加載器,有自己的獨立的類名稱空間。比較兩個類是否”相等“的前提是它們是由同一個類加載加載才有意義,即ClassLoader如果不同,兩個類必定不等。這樣使得在一個JVM中加載同一個模塊的不同版本的jar成為現實,基於反射功能,我們同樣可以很輕松實現不同版本的模塊測試。本文后面會提供簡單demo的實現。 - 代碼加密
沒有做過,想必是對class文件進行混淆、壓縮、native等等手段后的解密過程,這類需求還沒遇過。 - OSGi
是動態模型形同,在eclipse中插件的實現就是基於OSGi思想,而eclipse主要的應用就是插件,所以可以理解為eclipse插件是OSGi的應用典范。做的不多,僅限於了解。 - 熱部署
不停止服務,動態替換目標文件。ClassLoader動態加載jar包,如果做一個工程化的東西可能會費些周章,但是原理並不復雜。 - ...
總之,ClassLoader很重要,Java世界需要它。
功能測試小樣
本人在本地生成了test1.jar和test2.jar兩個jar包。這兩個jar都有類com.array7.jvm.classloader.Target
,此Demo要實現的是同時將這兩個jar包的同名類加載到JVM並且各自執行。
** test1.jar Target.java **
package com.array7.jvm.classloader;
public class Target {
public static void main(String[] args) {
System.out.print("test1");
}
}
** test2.jar Target.java **
package com.array7.jvm.classloader;
public class Target {
public static void main(String[] args) {
System.out.print("test2");
}
}
** TestDriver**
public static void main(String[] args) throws MalformedURLException, ClassNotFoundException, IllegalArgumentException, SecurityException, IllegalAccessException, NoSuchMethodException, InvocationTargetException {
ClassLoader loader1 = new URLClassLoader(new URL[]{new URL("file:/home/liushijie/workspace/test/out/artifacts/test1/test1.jar")}, TestDriver.class.getClassLoader());
ClassLoader loader2 = new URLClassLoader(new URL[]{new URL("file:/home/liushijie/workspace/test/out/artifacts/test2/test2.jar")}, TestDriver.class.getClassLoader());
String className = "com.array7.jvm.classloader.Target";
// loader1
System.out.print("test1.jar \t");
Class clazz1 = Class.forName(className, true, loader1);
clazz1.getMethod("main", String[].class).invoke(null, (Object) null);
System.out.println();
// loader2
System.out.print("test2.jar \t");
Class clazz2 = Class.forName(className, true, loader2);
clazz2.getMethod("main", String[].class).invoke(null, (Object) null);
System.out.println();
System.out.println("實例化后是否相等:" + clazz1.equals(clazz2));
}
輸出
test1.jar test1
test2.jar test2
實例化后是否相等:false
其他未提知識點
- ClassLoader的層級關系
- 雙親委托與打破
- 自定義ClassLoader