網絡類加載器
背景
由於在深入jvm虛擬機中看到了有部分說道class可以通過網絡的方式加載,於是就想到了是不是可以通過在網絡上發布jar包,然後程序動態加載網絡上的jar包(可拓展為熱更新)
代碼地址
調用模塊 https://coding.net/u/mich/p/easytry/git/tree/master/src/com/netclassloader
實現模塊 https://coding.net/u/mich/p/easytry/git/tree/master/netlogicImpl
接口模塊 https://coding.net/u/mich/p/easytry/git/tree/master/netlogicInterface
說明
該內容主要由三部分組成
- 接口模塊 主要包含接口jar包
- 實現模塊 導入接口jar包,然後實現接口的具體邏輯,打包成jar包,最後通過工具包,生成對應的接口文件,將jar包和接口文件都上傳至網絡,我這裏就直接使用了git上的地址,這裏還有生成了接口協議文件,類似於雙方協議,實現類和調用者的對應關系
- 調用模塊 自定義一個管理類,下載接口文件與接口jar包,解析jar包中的所有有效class文件,然後將協議文件以及類與class的map存放在成員變量中,當網絡加載器加載類指定類時,通過剛才的map獲取class文件的byte[],定義該類,管理類同時提供一個獲取服務的方法,通過協議文件,將接口與實現類綁定在一起
內容
接口模塊
接口模塊相對來說比較簡單,兩個接口
package com.netresource.logic; /** * Created by Mich on 2017/7/22. */ public interface ISayHello { Object sayHello(); }
package com.netresource.logic; /** * Created by Mich on 2017/7/22. */ public interface ISayWorld { Object sayWorld(); }
實現模塊
實現模塊需要先導入剛才接口模塊,然後寫兩個接口的實現類
package com.netlogic; import com.netresource.logic.ISayHello; /** * Created by Mich on 2017/7/22. */ public class SayHelloImpl implements ISayHello { @Override public Object sayHello() { return "Hello"; } }
package com.netlogic; import com.netresource.logic.ISayWorld; /** * Created by Mich on 2017/7/22. */ public class SayWorldImpl implements ISayWorld { @Override public Object sayWorld() { return "World"; } }
你會發現還有兩個util,fileutil主要是我一直使用的對文件的一些處理比較方便,就直接引用了,PropertiesUtil主要是協議文件的生成工具,通過傳入實現類的包名,然後掃描該包下的所有類把接口和實現做對應關系存放在output.properties文件中,需要註意的是,如果一個接口並不支持有多個實現類,但是一個實現類實現多個接口是可以的
最後將實現類打jar包,以及將協議文件上傳至網絡,我這裏圖方便直接放到了coding上https://coding.net/u/mich/p/easytry/git/raw/master/src/com/netclassloader/output/netlogicImpl.jar
https://coding.net/u/mich/p/easytry/git/raw/master/src/com/netclassloader/output/output.properties
調用模塊
同樣調用模塊需要引用接口模塊的jar包,然後介紹一下調用模塊的具體目錄結構(這裏無視output文件夾,這主要是我剛才上傳的兩個文件,jar包和協議文件,為了在一個模塊裏才放這裏,實際不需要這個文件夾)
NetClassManager,由於代碼有點多就不在這裏顯示了,具體可以再coding上去看,這裏主要介紹一下結構
- 構造函數,需要傳入兩個網絡地址,一個是協議文件網絡地址,一個是jar包的網絡地址
- getService通過傳入ClassLoader和Class,會根據協議文件的對應關系獲得網絡jar包上的對應實現類
- initProperties初始化協議文件,主要是從網絡下載協議文件,然後存放在成員變量中
- initImplJar初始化jar包,將jar包暫存在本地,然後解析獲得各個內部的class的byte[],通過協議文件,獲取所需要的類,存放在classMap成員變量中
- getClassMap外部獲取對應map,主要是給自定義的classLoader可以獲得指定的class的byte[]
NetClassLoader類加載器
package com.netclassloader; /** * Created by Mich on 17/7/17. */ public class NetClassLoader extends ClassLoader { private NetClassManager netClassManager; public NetClassLoader(NetClassManager netClassManager) { this.netClassManager = netClassManager; } @Override protected Class<?> findClass(String name) throws ClassNotFoundException { byte[] bytes = netClassManager.getClassMap().get(name); return defineClass(name, bytes, 0, bytes.length); } }
這個就比較簡單了,繼承了ClassLoader,構造函數傳入剛剛說明的管理類NetClassManager,然後重寫了findClass的方法,通過管理類的map來獲取字節數組
Main作為測試的入口函數
public static void main(String[] args) {
String jarUrl = "https://coding.net/u/mich/p/easytry/git/raw/master/src/com/netclassloader/output/netlogicImpl.jar";
String propertiesUrl = "https://coding.net/u/mich/p/easytry/git/raw/master/src/com/netclassloader/output/output.properties";
NetClassManager netClassManager = new NetClassManager(propertiesUrl, jarUrl);
NetClassLoader classLoader = new NetClassLoader(netClassManager);
ISayHello hello = netClassManager.getService(classLoader, ISayHello.class);
ISayWorld world = netClassManager.getService(classLoader, ISayWorld.class);
System.out.println(hello.sayHello());
System.out.println(world.sayWorld());
}
最後就是比較簡單的調用了,運行結果
最後
其實這裏也只是拋磚,如果具體使用,應該還需要版本控制,可以有一個專門的類,或者手動調用,去獲取最新的jar包,和最新的協議接口文件,當然更好一點可以再添加一個實現類的版本控制,這樣就需要修改的粒度更小了。對了如果要添加動態更新還需要修改NetClassLoader類,現在目前只是第一次加載處理,如果有更新,就需要重寫loadClass方法了。最後想想其實協議文件,和版本控制可以直接放在jar包中。。。等下次再繼續改進吧。。。
網絡類加載器