09. 类加载器命名空间

命名空间

每个类加载器都有自己的命名空间,命名空间由该加载器及所有父加载器所加载的类组成

在同一个命名空间中,不会出现类的完整名字(包括类的包名)相同的两个类

在不同的命名空间中,有可能会出现类的完整名字(包括类的包名)相同的两个类

关于命名空间,看几个例子

首先分别定义MyCat、MySample、MyClassLoader类

package com.zj.study.jvm.classloader;

public class MyCat {

    public MyCat() {
        System.out.println("MyCat is loaded by: " + this.getClass().getClassLoader());
    }
}
package com.zj.study.jvm.classloader;

public class MySample {

    public MySample() {
        System.out.println("MySample is loaded by: " + this.getClass().getClassLoader());
            // 会由加载MySample的类加载器尝试加载MyCat
        new MyCat();
    }
}
package com.zj.study.jvm.classloader;

import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.InputStream;

public class MyClassLoader extends ClassLoader{

    private String classLoaderName;

    private final String fileExtension = ".class";

    private String path;

    public void setPath(String path) {
        this.path = path;
    }

    public MyClassLoader(String classLoaderName) {
        super();
        this.classLoaderName = classLoaderName;
    }

    public MyClassLoader(ClassLoader classLoader, String classLoaderName) {
        super(classLoader);
        this.classLoaderName = classLoaderName;
    }

    @Override
    protected Class<?> findClass(String className) throws ClassNotFoundException {
        // 这两个信息输出 说明findClass()方法被调用
        System.out.println("findClass invoked: " + className);
        System.out.println("class loader name: " + this.classLoaderName);

        byte[] data = this.loadClassData(className);
        return this.defineClass(className, data, 0, data.length);
    }

    private byte[] loadClassData(String className) {
        InputStream is = null;
        byte[] data = null;
        ByteArrayOutputStream baos = null;

        // windows 系统 替换成\\
        className = className.replace(".", "/");

        try {
            is = new FileInputStream(new File( this.path + className + this.fileExtension));
            baos = new ByteArrayOutputStream();
            int ch = 0;
            while (-1 != (ch = is.read())){
                baos.write(ch);
            }
            data = baos.toByteArray();
        }catch (Exception e) {
            e.printStackTrace();
        }finally {
            try {
                is.close();
                baos.close();
            }catch (Exception e) {
                e.printStackTrace();
            }
        }
        return data;
    }

    @Override
    public String toString() {
        return "[" + this.classLoaderName + "]";
    }

}

上述代码很简单,MyClassLoader是自定义的一个类加载器,在MySample的构造方法中new了一个MyCat的实例

接着看测试代码

public class MyTest17 {
    public static void main(String[] args) throws ClassNotFoundException, IllegalAccessException, InstantiationException {
        MyClassLoader myClassLoader = new MyClassLoader("myClassLoader");
        myClassLoader.setPath("/Users/zj/Desktop/");
        Class<?> clazz = myClassLoader.loadClass("com.zj.study.jvm.classloader.MySample");
        System.out.println("clazz:" + clazz.hashCode());
        // 如果注释掉该行,那么并不会实例化MySample对象,即MySample构造方法不会被调用,因此不会实例化MyCat对象,即没有对MyCat进行主动使用
        Object object = clazz.newInstance();
    }
}

这里将classpath路径下的class文件复制到/Users/zj/Desktop/这个目录,然后删除classpath路径下的MyCat.class文件,运行上述测试代码会抛ClassNotFoundException异常,这是因为

在MySample的构造方法中,实例化了一个MyCat对象,需要加载、连接并初始化MyCat,此时会由加载MySample的类加载器尝试加载MyCat

显而易见,MySample是由系统类加载器加载的,而系统类加载器加载不了MyCat.class,因为删除了,所以会抛ClassNotFoundException异常

然后同样的代码,重新build一下项目,将classpath路径下的class文件复制到/Users/zj/Desktop/这个目录,然后删除classpath路径下的MySample.class文件,运行上述测试代码,不会抛异常,这是因为

MySample是有自定义类加载器myClassLoader加载的,myClassLoader会委托系统类加载器加载MyCat,是可以加载到的。

接下来修改MyCat.java

public class MyCat {

    public MyCat() {
        System.out.println("MyCat is loaded by: " + this.getClass().getClassLoader());
        System.out.println("from MyCat:" + MySample.class);
    }
}

同样的测试代码,重新build一下项目,将classpath路径下的class文件复制到/Users/zj/Desktop/这个目录,然后删除classpath路径下的MySample.class文件,运行上述测试代码会抛ClassNotFoundException异常,这是因为

对于类加载器命名空间,有如下重要说明

  • 子加载器所加载的类能够访问到父加载器所加载的类
  • 父加载器所加载的类无法访问到子加载器所加载的类

MySample类是由自定义类加载器myClassLoader加载的,而MyCat是由系统类加载器加载的,在MyCat的构造函数中,找MySample.class 是找不到的,因为系统类加载器是myClassLoader的父加载器,父加载器所加载的类无法访问到子加载器所加载的类

接下来再次修改MySample.java和MyCat.java

package com.zj.study.jvm.classloader;

public class MySample {

    public MySample() {
        System.out.println("MySample is loaded by: " + this.getClass().getClassLoader());
        // 会由加载MySample的类加载器尝试加载MyCat
        new MyCat();
        System.out.println("from MySample: " + MyCat.class);
    }
}
package com.zj.study.jvm.classloader;

public class MyCat {

    public MyCat() {
        System.out.println("MyCat is loaded by: " + this.getClass().getClassLoader());
    }
}

同样的测试代码,重新build一下项目,将classpath路径下的代码复制到/Users/zj/Desktop/这个目录,然后删除classpath路径下的MySample.class文件,运行上述测试代码不会抛ClassNotFoundException异常,这是因为

MySample类是由自定义类加载器myClassLoader加载的,而MyCat是由系统类加载器加载的,在MySample的构造函数中,找MyCat.class 是找的到的,因为myClassLoader是系统类加载器的子加载器,子加载器所加载的类能够访问到父加载器所加载的类

不同类加载器的命名空间关系

同一个命名空间的类是相互可见的

子加载器的命名空间包含所有父加载器的命名空间。因此由子加载器加载的类能看见父加载器加载的类。例如系统类加载器加载的类能看见根类加载器加载的类。

由父加载器加载的类不能看见子加载器加载的类

如果两个加载器之间没有直接或间接的父子关系,那么它们各自加载的类相互不可见

试验1

public class MyTest20 {
    // myClassLoader1、myClassLoader2是两个MyClassLoader对象
    // myClassLoader1、myClassLoader2调用loadClass分别加载相同的binary name
    // 通过反射创建两个实例
    // 通过反射的方式 调用setMyPerson()方法
    public static void main(String[] args) throws ClassNotFoundException, IllegalAccessException, InstantiationException, NoSuchMethodException, InvocationTargetException {
        MyClassLoader myClassLoader1 = new MyClassLoader("loader1");
        MyClassLoader myClassLoader2 = new MyClassLoader("loader2");

        Class<?> clazz1 = myClassLoader1.loadClass("com.zj.study.jvm.classloader.MyPerson");
        Class<?> clazz2 = myClassLoader2.loadClass("com.zj.study.jvm.classloader.MyPerson");

        // 结果为true
        System.out.println(clazz1 == clazz2);

        Object object1 = clazz1.newInstance();
        Object object2 = clazz2.newInstance();

        Method method = clazz1.getMethod("setMyPerson", Object.class);
        // 在object1上调用method方法,参数是object2
        method.invoke(object1, object2);
    }
}

试验2

把MyPerson.class复制到桌面,删除classpath路径下MyPerson.class文件

public class MyTest21 {
    // myClassLoader1、myClassLoader2是两个MyClassLoader对象
    // myClassLoader1、myClassLoader2调用loadClass分别加载相同的binary name
    // 通过反射创建两个实例
    // 通过反射的方式 调用setMyPerson()方法
    public static void main(String[] args) throws ClassNotFoundException, IllegalAccessException, InstantiationException, NoSuchMethodException, InvocationTargetException {
        MyClassLoader myClassLoader1 = new MyClassLoader("loader1");
        MyClassLoader myClassLoader2 = new MyClassLoader("loader2");
        myClassLoader1.setPath("/Users/zj/Desktop/");
        myClassLoader2.setPath("/Users/zj/Desktop/");

        Class<?> clazz1 = myClassLoader1.loadClass("com.zj.study.jvm.classloader.MyPerson");
        Class<?> clazz2 = myClassLoader2.loadClass("com.zj.study.jvm.classloader.MyPerson");

        // 结果为false
        System.out.println(clazz1 == clazz2);

        Object object1 = clazz1.newInstance();
        Object object2 = clazz2.newInstance();

        // clazz1 和 clazz2 不可见, 由此生成的对象 object1 和 object2 不可见
        Method method = clazz1.getMethod("setMyPerson", Object.class);
        // 在object1上调用method方法,参数是object2
        method.invoke(object1, object2);
    }
}
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 199,393评论 5 467
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 83,790评论 2 376
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 146,391评论 0 330
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 53,703评论 1 270
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,613评论 5 359
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,003评论 1 275
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,507评论 3 390
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,158评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,300评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,256评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,274评论 1 328
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,984评论 3 316
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,569评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,662评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,899评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,268评论 2 345
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 41,840评论 2 339

推荐阅读更多精彩内容