1. 程式人生 > >網絡類加載器

網絡類加載器

str 文件中 png spa class ref 代碼 rop 部分

背景

由於在深入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

說明

該內容主要由三部分組成

  1. 接口模塊  主要包含接口jar包
  2. 實現模塊   導入接口jar包,然後實現接口的具體邏輯,打包成jar包,最後通過工具包,生成對應的接口文件,將jar包和接口文件都上傳至網絡,我這裏就直接使用了git上的地址,這裏還有生成了接口協議文件,類似於雙方協議,實現類和調用者的對應關系
  3. 調用模塊   自定義一個管理類,下載接口文件與接口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上去看,這裏主要介紹一下結構

技術分享

  1. 構造函數,需要傳入兩個網絡地址,一個是協議文件網絡地址,一個是jar包的網絡地址
  2. getService通過傳入ClassLoader和Class,會根據協議文件的對應關系獲得網絡jar包上的對應實現類
  3. initProperties初始化協議文件,主要是從網絡下載協議文件,然後存放在成員變量中
  4. initImplJar初始化jar包,將jar包暫存在本地,然後解析獲得各個內部的class的byte[],通過協議文件,獲取所需要的類,存放在classMap成員變量中
  5. 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包中。。。等下次再繼續改進吧。。。

網絡類加載器