类加载器的双亲委派及打破双亲委派

超人汪小建(seaboat) 超人汪小建(seaboat)     2022-07-28     733

关键词:

一般的场景中使用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();
        }
    }

}

jvm——类加载器的双亲委派模型

类加载器双亲委派模型,如下图所示:双亲委派模型的工作过程  如果一个类加载器收到了类加载的请求,它首先不会自己去尝试加载这个类,而是把这个请求委派给父类加载器去完成,每一个层次的类加载器都是如此,因此... 查看详情

java双亲委派模型:为什么要双亲委派?如何打破它?破在哪里?(代码片段)

文章目录一、前言二、类加载器三、双亲委派机制1、什么是双亲委派2、为什么要双亲委派?四、破坏双亲委派五、Class.forName默认使用的类加载器六、线程上下文类加载器七、要点回顾一、前言平时做业务开发比较少接触类... 查看详情

为什么说spi打破双亲委派机制

...机制简单介绍ClassLoader的双亲委派机制:java类通过Classloader加载,Classloader之间有继承关系,AppClassLoader继承ExtClassloader继承BootstrapClassloader。在类加载时,子加载器会调用父加载器来加载类,如果父加载器不能加载... 查看详情

jdbc是如何打破双亲委派模式的(代码片段)

...什么JDBC需要打破双亲委派机制JDBC的DriverManager与SPI机制类加载的机制以及双亲委派机制的介绍可以参考JVM类加载机制在JDBC4.0之后,我们不再需要调用Class.https://juejin.cn/post/7007292903361871903为什么JDBC需要打破双亲委派机制JDBC的D... 查看详情

java双亲委派机制&与打破

简介当某个类加载器需要加载某个.class文件时,它首先把这个任务委托给他的上级类加载器,递归这个操作,如果上级的类加载器没有加载,自己才会去加载这个类。参考链接https://www.jianshu.com/p/1e4011617650类加载器类别BootstrapClassL... 查看详情

code皮皮虾带你盘点双亲委派机制原理优缺点,以及如何打破它?(代码片段)

...点打破双亲委派机制?前提知识:线程上下文类加载器双亲委派出现之前JDBC打破双亲委派机制Tomcat如何打破双亲委派机制?1.自定义类加载器2.使用线程上下文类加载器💖福利🌊Java入门到就业学习路线规划&#x... 查看详情

code皮皮虾带你盘点双亲委派机制原理优缺点,以及如何打破它?(代码片段)

...点打破双亲委派机制?前提知识:线程上下文类加载器双亲委派出现之前JDBC打破双亲委派机制Tomcat如何打破双亲委派机制?1.自定义类加载器2.使用线程上下文类加载器💖福利🌊Java入门到就业学习路线规划&#x... 查看详情

双亲委派机制

类加载器双亲委派模型图双亲委派模型构成启动类加载器,扩展类加载器,应用程序类加载器,自定义类加载器双亲委派模型工作过程是如果一个类加载器收到类加载的请求,它首先不会自己去尝试加载这个类,而是把这个请求... 查看详情

为什么说spi打破双亲委派机制

...机制简单介绍ClassLoader的双亲委派机制:java类通过Classloader加载,Classloader之间有继承关系,AppClassLoader继承ExtClassloader继承BootstrapClassloader。在类加载时,子加载器会调用父加载器来加载类,如果父加载器不能加载... 查看详情

从源码理解双亲委派机制,原来如此简单(代码片段)

本文,我们介绍类加载机制的几个硬核问题:1从JDK源码级别剖析双亲委派机制原理2自定义类加载器研究如何打破双亲委派机制3.理解Tomcat的沙箱安全机制目录1双亲委派机制介绍2双亲委派机制实现原理3.打破双亲委派机制... 查看详情

tomcat如何打破双亲委派机制实现隔离web应用的?(代码片段)

Tomcat通过自定义类加载器WebAppClassLoader打破双亲委派,即重写了JVM的类加载器ClassLoader的findClass方法和loadClass方法,以优先加载Web应用目录下的类。Tomcat负责加载我们的Servlet类、加载Servlet所依赖的JAR包。Tomcat本身也是个Ja... 查看详情

双亲委派

类加载器将.class文件加载到JVM,首先是看当前类是不是使用自定义加载类加载的,如果不是,就委派应用类加载器加载,如果有加载过这个class文件,那就不用再加载了。如果没有,那么会拿到父加载器... 查看详情

什么类加载器的双亲委托模型?(代码片段)

类加载器的双亲委托模型并不是一个强制的约束模型,而是Java设计者推荐给开发者的一种加载器方式。上面类加载器的父子关系一般不会以继承的方式实现,而是采用组合的关系来复用父类加载器的代码。工作过程:如果一个... 查看详情

阿里预面:谈谈你对双亲委派机制的理解?这个名字有啥问题?如何打破?为啥双亲委派?...(代码片段)

...c;运行程序肯定先执行main方法,要执行main方法就得先加载其类,也就是java.lang.String,类加载请求来到类加载器,会一层一层往上委派,最终来到引导/启动类加载器(BootstrapClass 查看详情

双亲委派机制(代码片段)

双亲委派机制双亲委派的原理:如果一个类加载器收到了类加载的请求,它首先不会自己去尝试加载这个类,而是把这个请求委托给父类加载器去完成,每一个层次的类加载器都是如此,因此所有的加载请求最终都应该传送到顶... 查看详情

双亲委派模型

双亲委派模型的概念    如果一个类加载器收到了类加载的请求,它首先不会自己去尝试加载这个类,而是把这个请求委派给父类加载器去完成,每一个层次的加载器都是如此,因为所有的类请求都会传给顶层的启... 查看详情

关于类加载

类加载器种类  双亲委派模型所谓双亲委派是指每次收到类加载请求时,先将请求委派给父类加载器完成(所有加载请求最终会委派到顶层的BootstrapClassLoader加载器中),如果父类加载器无法完成这个加载(该加载器的搜... 查看详情

类加载器和双亲委派

  这张图清晰吧类加载器的作用不仅仅是实现类的加载,它还与类的的“相等”判定有关,关系着Java“相等”判定方法的返回结果,只有在满足如下三个类“相等”判定条件,才能判定两个类相等。1、两个类来自同一个Class... 查看详情