JAVA类加载器分析

JVM三种预定义类型类加载器:

启动(Bootstrap)类加载器:引导类装入器是用本地代码实现的类装入器,它负责将 <Java_Runtime_Home>/lib 下面的类库加载到内存中,但是虚拟机出于安全等因素考虑,不会加载< Java_Runtime_Home >/lib存在的陌生类,开发者通过将要加载的非JDK自身的类放置到此目录下期待启动类加载器加载是不可能的

标准扩展(Extension)类加载器:扩展类加载器是由 Sun ExtClassLoadersun.misc.Launcher$ExtClassLoader 实现的。它负责将 < Java_Runtime_Home >/lib/ext 或者由系统变量 java.ext.dir 指定位置中的类库加载到内存中

系统(System)类加载器:系统类加载器是由 Sun AppClassLoadersun.misc.Launcher$AppClassLoader)实现的。

 

类加载双亲委派机制:

JVM在加载类时默认采用的是双亲委派机制。通俗的讲,就是某个特定的类加载器在接到加载类的请求时,首先将加载任务委托给父类加载器,依次递归,如果父类加载器可以完成类加载任务,就成功返回;只有父类加载器无法完成此加载任务时,才自己去加载。关于虚拟机默认的双亲委派机制,

ClassLoader主要是通过loadClass()来加载类的。

loadClass()方法的代码:

// 首先判断该类型是否已经被加载

        Class c = findLoadedClass(name);

        if (c == null) {

            //如果没有被加载,就委托给父类加载或者委派给启动类加载器加载

            try {

                if (parent != null) {

//如果存在父类加载器,就委派给父类加载器加载

                    c = parent.loadClass(name, false);

                } else {

//如果不存在父类加载器,就检查是否是由启动类加载器加载的类,通过调用本地方法native Class findBootstrapClass(String name)

                    c = findBootstrapClass0(name);

                }

            } catch (ClassNotFoundException e) {

        // 如果父类加载器和启动类加载器都不能完成加载任务,才调用自身的加载功能

                c = findClass(name);

            }

        }

通过分析上面代码,可以得出类加载器的加载顺序:

BootstrapClassLoader-----ExtensionClassLoader----AppClassLoader---自定义的类加载器

怎么知道当前类使用了什么类加载器呢?

System.out.println(this.getClass().getClassLoader());----得到当前的类加载器

 

System.out.println(this.getClass().getClassLoader().getParent());----得到当前类的父加载器

System.out.println(this.getClass().getClassLoader().getParent().getParent());----得到当前类的父加载器的父加载器

 

假如当前类的加载器为sun.misc.Launcher$AppClassLoader@197d257,那么它的父加载器则为

sun.misc.Launcher$ExtClassLoader@7259da,但是扩展类加载器的父加载器则为NULL

 

扩展类加载器的父加载器为什么会为NULL呢?

由于引导类加载器涉及到虚拟机本地实现细节,开发者无法直接获取到启动类加载器的引用,所以不允许直接通过引用进行操作。

 

扩展类加载器的父加载器怎么使父加载器为NULL的呢?

 扩展类加载器(ExtClassLoader调用ClassLoader(ClassLoader parent)构造函数将父类加载器设置为null

 

但是我们可以分析前面的代码。。当父加载器为NULL,JAVA会调用启动类加载器(findBootstrapClass0(name)

如何实现我们的自定义类加载器呢?

实现自定义加载器其它很简单。只要实现ClassLoader的findClass()方法就行了。

大概如下:

findClass(String name){

byte[] buffer=//读取文件的字节数组

return defineClass(buffer,0,buffer.length);

}

当自定义类加载器加载的类有继承相应的类或实现相应接口的话,必须把相应的父类或接口拷贝到同一路径。因为类加载器在加载子类时,会同时加载父类。

关于类的转型问题:

由于类加载器是相互隔离的,当classLoaderA加载了A类,classLoaderB也加载了A类。这个类对象是不能互相转型的。会报ClassCastException异常。

假如自定义类加载器加载A类时,而A类又实现了接口B时,我们是可以将A类的实例转成B接口的。原因是类加载器在加载A类时,会同时加载A类的所有父类(包括接口).

JAVA类加载器采用了委托模式。当一个类对象己经被加载了,就不会在去加载该类。还是直接返回。那么如何重新加载这个类呢?

实现自定义的类加载器,当类加载完成后。调用相应方法,然后在去掉相应类加载器的引用。这样类不会在存在类加载器中了。

比如:classLoaderA=null(去掉A加载器的引用)

你可能感兴趣的:(java,jvm,虚拟机,ext,sun)