將實現類通過網路的方式下載下來,以達到動態載入的目的 git地址:https://coding.net/u/mich/p/easytry/git/tree/master/src/com/netclassloader ...
背景
由於在深入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包中。。。等下次再繼續改進吧。。。