国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 編程 > Java > 正文

java 詳解類加載器的雙親委派及打破雙親委派

2019-11-26 13:15:58
字體:
來源:轉載
供稿:網友

java 詳解類加載器的雙親委派及打破雙親委派

一般的場景中使用Java默認的類加載器即可,但有時為了達到某種目的又不得不實現自己的類加載器,例如為了達到類庫的互相隔離,例如為了達到熱部署重加載功能。這時就需要自己定義類加載器,每個類加載器加載各自的類庫資源,以此達到資源隔離效果。在對資源的加載上可以沿用雙親委派機制,也可以打破雙親委派機制。

一、沿用雙親委派機制自定義類加載器很簡單,只需繼承ClassLoader類并重寫findClass方法即可。如下例子:

①先定義一個待加載的類Test,它很簡單,只是在構建函數中輸出由哪個類加載器加載。

public class Test {  public Test(){    System.out.println(this.getClass().getClassLoader().toString());  }}

②定義一個TestClassLoader類繼承ClassLoader,重寫findClass方法,此方法要做的事情是讀取Test.class字節流并傳入父類的defineClass方法即可。然后就可以通過自定義累加載器TestClassLoader對Test.class進行加載,完成加載后會輸出“TestLoader”。

public class TestClassLoader extends ClassLoader {  private String name;  public TestClassLoader(ClassLoader parent, String name) {    super(parent);    this.name = name;  }  @Override  public String toString() {    return this.name;  }  @Override  public Class<?> findClass(String name) {    InputStream is = null;    byte[] data = null;    ByteArrayOutputStream baos = new ByteArrayOutputStream();    try {      is = new FileInputStream(new File("d:/Test.class"));      int c = 0;      while (-1 != (c = is.read())) {        baos.write(c);      }      data = baos.toByteArray();    } catch (Exception e) {      e.printStackTrace();    } finally {      try {        is.close();        baos.close();      } catch (IOException e) {        e.printStackTrace();      }    }    return this.defineClass(name, data, 0, data.length);  }  public static void main(String[] args) {    TestClassLoader loader = new TestClassLoader(        TestClassLoader.class.getClassLoader(), "TestLoader");    Class clazz;    try {      clazz = loader.loadClass("test.classloader.Test");      Object object = clazz.newInstance();    } catch (Exception e) {      e.printStackTrace();    }   }}

二、打破雙親委派機制則不僅要繼承ClassLoader類,還要重寫loadClass和findClass方法,如下例子:

①定義Test類。

public class Test {  public Test(){    System.out.println(this.getClass().getClassLoader().toString());  }}

②重新定義一個繼承ClassLoader的TestClassLoaderN類,這個類與前面的TestClassLoader類很相似,但它除了重寫findClass方法外還重寫了loadClass方法,默認的loadClass方法是實現了雙親委派機制的邏輯,即會先讓父類加載器加載,當無法加載時才由自己加載。這里為了破壞雙親委派機制必須重寫loadClass方法,即這里先嘗試交由System類加載器加載,加載失敗才會由自己加載。它并沒有優先交給父類加載器,這就打破了雙親委派機制。

public class TestClassLoaderN extends ClassLoader {  private String name;  public TestClassLoaderN(ClassLoader parent, String name) {    super(parent);    this.name = name;  }  @Override  public String toString() {    return this.name;  }  @Override  public Class<?> loadClass(String name) throws ClassNotFoundException {    Class<?> clazz = null;    ClassLoader system = getSystemClassLoader();    try {      clazz = system.loadClass(name);    } catch (Exception e) {      // ignore    }    if (clazz != null)      return clazz;    clazz = findClass(name);    return clazz;  }  @Override  public Class<?> findClass(String name) {    InputStream is = null;    byte[] data = null;    ByteArrayOutputStream baos = new ByteArrayOutputStream();    try {      is = new FileInputStream(new File("d:/Test.class"));      int c = 0;      while (-1 != (c = is.read())) {        baos.write(c);      }      data = baos.toByteArray();    } catch (Exception e) {      e.printStackTrace();    } finally {      try {        is.close();        baos.close();      } catch (IOException e) {        e.printStackTrace();      }    }    return this.defineClass(name, data, 0, data.length);  }  public static void main(String[] args) {    TestClassLoaderN loader = new TestClassLoaderN(        TestClassLoaderN.class.getClassLoader(), "TestLoaderN");    Class clazz;    try {      clazz = loader.loadClass("test.classloader.Test");      Object object = clazz.newInstance();    } catch (Exception e) {      e.printStackTrace();    }  }}

感謝閱讀,希望能幫助到大家,謝謝大家對本站的支持!

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 长沙市| 米林县| 西昌市| 哈密市| 连南| 安泽县| 合山市| 聂荣县| 高雄县| 霍林郭勒市| 鄂托克前旗| 庄河市| 冀州市| 常熟市| 广灵县| 洪洞县| 宁河县| 确山县| 牟定县| 炉霍县| 法库县| 庆云县| 吉林市| 政和县| 永福县| 渭源县| 北川| 宝鸡市| 什邡市| 汽车| 中江县| 锦州市| 谷城县| 新余市| 泰顺县| 清涧县| 张家川| 辉南县| 长春市| 彭水| 清涧县|