美文网首页
类加载器的命名空间 实例验证

类加载器的命名空间 实例验证

作者: rock_fish | 来源:发表于2019-11-01 10:40 被阅读0次

所属文集:ClassLoader串烧


同一个命名空间中,类只加载一份

AppClassLoader加载程序中自定义的类。无论加载多少次,只要是被AppClassLoader加载的,其Class信息hashcode都是相同的。

子加载器可见父加载器加载的类

到处都是例子。比如核心类库的类,AppClassLoader所加载的类,都能使用String

父加载器不可见子加载器所加载的类,只能验证加载不到

应用中有Parent类,Son类。Parent类中有个getSon方法中,使用Son类。
编译后,在classpath下,移除Son类到子加载器的加载目录中,让AppClassLoader无法加载。
达到我们的运行环境要求:Parent由父加载器加载,Son由子加载器加载.
调用Parent的getSon方法的时候要new Son类,会触发Son类的加载,但是加载不到Son类,报错信息为:java.lang.NoClassDefFoundError: com/rock/Son

public class Parent {

    private Object son;

    public Object getSon(){
        return new Son();
    }
    public Object setSon(Object son){
        this.son = son;
        return this.son;
    }
}  
public class Son {
}
/**
     * 父加载器加载不了,子加载器所加载的类,
     * 父加载器加载Parent
     * 子加载器加载son
     * Parent#getSon 方法里 new Son()对象.//报错,找不到Son的类定义.
     */


    @Test
    public void testParentCanntFindSon(){
        CustomClassLoader01 customClassLoader01 = new CustomClassLoader01(ClassLoader.getSystemClassLoader());
        try {
            Class<?> classParent = customClassLoader01.loadClass("com.rock.Parent");
            System.out.println("classParent:" + classParent.getClassLoader());
            Class<?> classSon = customClassLoader01.loadClass("com.rock.Son");
            System.out.println("classSon:" + classSon.getClassLoader());
            Object objParent = classParent.newInstance();

            Object objSon = classSon.newInstance();
            
            Method setSon = classParent.getMethod("setSon",Object.class);
            Object resultSon = setSon.invoke(objParent, objSon);
            System.out.println(resultSon.getClass());
            System.out.println(resultSon.getClass().getClassLoader());

            try {
                Method getSon = classParent.getMethod("getSon");
                Object invoke = getSon.invoke(objParent);
                System.out.println(invoke.getClass().getClassLoader());
            }catch (Exception exp){
                //java.lang.NoClassDefFoundError: com/rock/Son
                exp.printStackTrace();
            }
        }catch (Exception exp){
            exp.printStackTrace();
        }
    }
不同命名空间的互相不可见

自定义类加载器的多个实例,分别加载Man类,并反射实例化对象,对象之间互相调用setFather赋值,会报错。a空间的不识别b空间的,即使全限定名相同;但也不识别。

public class Man {
    private Man father;
    public void setFather(Object obj){
        father = (Man)obj;
    }
}
 @Test
    public void testSifferentNamespaceClass(){
        CustomClassLoader01 customClassLoader01 = new CustomClassLoader01(ClassLoader.getSystemClassLoader());
        CustomClassLoader01 customClassLoader02 = new CustomClassLoader01(ClassLoader.getSystemClassLoader());
        try {
            Class<?> aClass1 = customClassLoader01.loadClass("com.rock.Man");
            System.out.println("class1 : " + aClass1.getClassLoader());
            System.out.println("class1 : " + aClass1.);

            Class<?> aClass2 = customClassLoader02.loadClass("com.rock.Man");
            System.out.println("class2 : " + aClass1.getClassLoader());
            System.out.println("class2 : " + aClass2);
            Object man1 = aClass1.newInstance();
            Object man2 = aClass2.newInstance();
            Method setFather = aClass1.getMethod("setFather", Object.class);
            setFather.invoke(man1,man2);


        }catch (Exception exp){
            exp.printStackTrace();
        }
    }
class1 : com.rock.classLoader.CustomClassLoader01@1f28c152
class1 : 2006034581
class2 : com.rock.classLoader.CustomClassLoader01@1f28c152
class2 : 488044861
...
Caused by: java.lang.ClassCastException: com.rock.Man cannot be cast to com.rock.Man
    at com.rock.Man.setFather(Man.java:6)
    ... 27 more

命名空间的设计实现

image.png

传送门:ClassLoader串烧

相关文章

  • 类加载器

    类加载过程 类加载器层次结构 使用类加载器作为命名空间 实现自己的类加载器 字节码验证

  • 类加载器的命名空间 实例验证

    所属文集:ClassLoader串烧 同一个命名空间中,类只加载一份 AppClassLoader加载程序中自定义...

  • 06_JVM学习笔记_类的命名空间、类的卸载

    类的命名空间 每个类加载器都有自己的命名空间,命名空间由该加载器及所有父加载器所加载的类组成。 在同一个命名空间中...

  • 【深入理解Java虚拟机 】类加载器的命名空间以及类的卸载

    类加载器的命名空间 每个类加载器又有一个命名空间,由其以及其父加载器组成 类加载器的命名空间的作用和影响 每个类加...

  • 09. 类加载器命名空间

    命名空间 每个类加载器都有自己的命名空间,命名空间由该加载器及所有父加载器所加载的类组成 在同一个命名空间中,不会...

  • 命名空间与类的卸载

    命名空间介绍  每个类加载器都有自己的命名空间,命名空间由该加载器及所有的父加载器所加载的类组成。也就是说,命名空...

  • JAVA ClassLoader命名空间

    每个类加载器都对应一个命名空间,命名空间由该加载器及所有父类加载器所加载的类组成。 特点: 在同一个命名空间中,不...

  • 类加载范围

    类加载范围  系统类加载器加载的为项目中的class 每个类加载器都有自己的命名空间,命名空间由该加载器及所有的父...

  • JVM-2

    命名空间的补充说明 关于命名空间的补充说明子加载器所加载的类可以访问父加载器所加载的类父加载器所加载的类无法访问子...

  • composer自动加载

    理解自动加载函数的原理 在实例话一个类时,如有全局的自动加载函数,则会将类的命名空间及类名传入到加载函数,然后由加...

网友评论

      本文标题:类加载器的命名空间 实例验证

      本文链接:https://www.haomeiwen.com/subject/aoynvctx.html