类加载器的双亲委派及打破双亲委派
一般的場景中使用Java默認的類加載器即可,但有時為了達到某種目的又不得不實現自己的類加載器,例如為了達到類庫的互相隔離,例如為了達到熱部署重加載功能。這時就需要自己定義類加載器,每個類加載器加載各自的類庫資源,以此達到資源隔離效果。在對資源的加載上可以沿用雙親委派機制,也可以打破雙親委派機制。
一、沿用雙親委派機制自定義類加載器很簡單,只需繼承ClassLoader類并重寫findClass方法即可。如下例子:
①先定義一個待加載的類Test,它很簡單,只是在構建函數中輸出由哪個類加載器加載。
public class Test {
? ? public Test(){
? ? ? ? System.out.println(this.getClass().getClassLoader().toString());
? ? }
}
1
2
3
4
5
6
7
②定義一個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();
? ? ? ? }?
? ? }
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
二、打破雙親委派機制則不僅要繼承ClassLoader類,還要重寫loadClass和findClass方法,如下例子:
①定義Test類。
public class Test {
? ? public Test(){
? ? ? ? System.out.println(this.getClass().getClassLoader().toString());
? ? }
}
1
2
3
4
5
②重新定義一個繼承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();
? ? ? ? }
? ? }
}
---------------------?
作者:超人汪小建(seaboat)?
來源:CSDN?
原文:https://blog.csdn.net/wangyangzhizhou/article/details/51787377?
總結
以上是生活随笔為你收集整理的类加载器的双亲委派及打破双亲委派的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: javac 编译与 JIT 编译
- 下一篇: 类加载机制-双亲委派,破坏双亲委派--这