欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

自定義類加載器以及打破雙親委派模型解析

 更新時(shí)間:2022年11月19日 10:22:15   作者:天瑕  
這篇文章主要介紹了自定義類加載器以及打破雙親委派模型解析,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教

1 自定義類加載器

自定義類加載器的代碼很簡(jiǎn)單,只需要繼承ClassLoader類,覆寫findClass方法即可

其默認(rèn)實(shí)現(xiàn)是會(huì)拋出一個(gè)異常:

import java.io.FileInputStream;
 
public class MyClassLoader extends ClassLoader {
 
    private String classPath;
 
    public MyClassLoader(String classPath) {
        this.classPath = classPath;
    }
 
    private byte[] loadByte(String name) throws Exception {
        name = name.replaceAll("\\.", "/");
        FileInputStream fis = new FileInputStream(classPath + "/" + name + ".class");
        int len = fis.available();
        byte[] data = new byte[len];
        fis.read(data);
        fis.close();
        return data;
    }
 
    @Override
    protected Class<?> findClass(String name) {
        byte[] data = new byte[0];
        try {
            data = loadByte(name);
        } catch (Exception e) {
            e.printStackTrace();
        }
        return defineClass(name, data, 0, data.length);
    }
}

這里是會(huì)讀取指定的類路徑classPath下的class文件。

相應(yīng)的測(cè)試代碼如下所示:

public class MyClassLoaderTest {
 
    public static void main(String[] args) throws Exception {
        MyClassLoader classLoader = new MyClassLoader("D:/test");
        Class clazz = classLoader.loadClass("com.hys.test.User");
        System.out.println(clazz.getClassLoader().getClass().getName());
    }
}

這里以User類代碼為例,將其class文件放到D:/test指定目錄下:

隨后需要注意的是,需要將當(dāng)前工作空間中的User.java文件刪除。

如果不刪除,根據(jù)雙親委派模型,該類會(huì)由AppClassLoader來加載,不會(huì)由自定義的的MyClassLoader來進(jìn)行加載,最后運(yùn)行測(cè)試代碼

結(jié)果如下:

2 打破雙親委派模型

在像一些Tomcat的源碼中,WebappClassLoader會(huì)打破雙親委派機(jī)制。這里我們也來簡(jiǎn)單模擬一下。

實(shí)現(xiàn)代碼依然很簡(jiǎn)單,只需要在上述MyClassLoader類中覆寫loadClass方法即可,如下:

@Override
protected Class<?> loadClass(String name, boolean resolve) throws ClassNotFoundException {
    synchronized (getClassLoadingLock(name)) {
        Class<?> c = findLoadedClass(name);
        if (c == null) {
            c = findClass(name);
        }
        if (resolve) {
            resolveClass(c);
        }
        return c;
    }
}

這里loadClass方法的代碼使用的是父類ClassLoader的源碼,然后把其中使用雙親委派的代碼刪掉,這樣MyClassLoader不用再向上去找類加載器,只會(huì)在本類中處理,這樣就打破了雙親委派模型。

然后因?yàn)檫\(yùn)行時(shí)需要加載Object類,所以將Object.class文件復(fù)制到D:/test目錄下

如下所示:

隨后再次運(yùn)行測(cè)試類

結(jié)果如下:

可以看到,java.lang包的代碼禁止被自定義的類加載器加載,防止核心API被篡改。

這是Java內(nèi)部的安全檢查機(jī)制。這里我們這種寫法是將所有的類都交由MyClassLoader來處理,所以無法加載Java核心的類庫,但是Tomcat中的類加載機(jī)制只是自定義的WebappClassLoader和CommonClassLoader打破了雙親委派模型,而其上面的BootstrapClassLoader、ExtensionClassLoader和AppClassLoader仍然還是會(huì)走雙親委派的,所以不會(huì)有問題。

以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。

相關(guān)文章

最新評(píng)論