Java中的ClassLoader

基本概念

ClassLoader 是一个重要的 Java 执行时系统组件,它负责在运行时查找和装入 Class 字节码文件。JVM 在运行时会产生三个ClassLoader:根装载器、扩展类装载器(ExtClassLoader)和系统类装载器(AppClassLoader)。其中,根装载器不是ClassLoader的子类,它使用C++编写,因此我们在 Java 中看不到它,根装载器负责装载JRE的核心类库,如JRE目标下的 rt.jar、charsets.jar 等。ExtClassLoader 和AppClassLoader 都是 ClassLoader 的子类。其中 ExtClassLoader 负责装载 JRE 目录 ext 中的 JAR 类包;AppClassLoader 负责装载 ClassPath 路径下的类包。

  • 根装载器,这个类加载器负责将存放在 <JAVA_HOME>\lib 目录中的。启动类加载器无法被 Java 程序直接引用,用户在编写自定义类加载器时,如果需要把加载请求委派给引导类加载器,那直接使用null代替即可。
  • 扩展类加载器(Extension ClassLoader),这个加载器由 sun.misc.Launcher$ExtClassLoader 实现,它负责加载 <JAVA_HOME>\lib\ext 目录中的,或者被 java.ext.dirs 系统变量所指定的路径中的所有类库,开发者可以直接使用扩展类加载器。
  • 应用程序类加载器(Application ClassLoader),这个类加载器由 sun.misc.Launcher$AppClassLoader 实现。由于这个类加载器是 ClassLoader 中的 getSystemClassLoader() 方法的返回值,所以一般也称它为系统类加载器。它负责加载用户类路径(ClassPath)上所指定的类库,开发者可以直接使用这个类加载器,如果应用程序中没有自定义自己的类加载器,一般情况下这个就是程序中默认的类加载器。

我们的应用程序都是由这3种类加载器互相配合进行加载的,如果有必要,还可以加入自己定义的类加载器。这些类加载器之间的关系一般为:

这三个类装载器之间存在父子层级关系,默认情况下,使用AppClassLoader装载应用程序的类,用以下代码证明:

package com.wlee.test;

public class ClassLoaderTest {

    public static void main(String[] args) {
        //Thread.currentThread():返回对当前正在执行的线程对象的引用。
        //getContextClassLoader():返回该线程的上下文ClassLoader。
        ClassLoader loader = Thread.currentThread().getContextClassLoader();
        System.out.println("当前的loader:" + loader);
        System.out.println("父级的loader:" + loader.getParent());
        System.out.println("祖父级的loader:" + loader.getParent().getParent());
    }
}

输出结果:

当前的loader:sun.misc.Launcher$AppClassLoader@18b4aac2
父级的loader:sun.misc.Launcher$ExtClassLoader@1d44bcfa
祖父级的loader:null

通过以上的输出信息,很明显 ClassLoader 是 AppClassLoader,父 ClassLoader 是 ExtClassLoader,祖父 ClassLoader 是根类装载器,因为在 Java 中无法获得它,所以返回 null。

另外,ClassLoader 的几个重要方法:

方法解释
Class defineClass(String name, byte[] b, int off, int len)将类文件的字节数组转换成JVM内部的java.lang.Class对象。字节数组可以从本地文件系统、远程网络获取。name为字节数组对应的全限定类名。
Class findSystemClass(String name)从本地文件系统载入Class文件,如果本地文件系统不存在该Class文件,将抛出ClassNotFoundException异常。该方法是JVM默认使用的装载机制。
ClassLoader getParent()获取类装载器的父装载器,除根装载器外,所有的类装载器都有且仅有一个父装载器,ExtClassLoader的父装载器是根装载器,因为根装载器非Java编写,所以无法获得,将返回null。

进阶 - 自定义ClassLoader

首先我们自定义一个简单的类,并编译层 class 文件,放在指定的目录下,比如 d:/test/ 目录下,类代码如下:

package com.wlee.test;

public class App {
    private String name = "测试App";
}

这个类编译之后把编译好的 class 文件放在 d:/test/ 目录下,也就是文件路径是 d:/test/App.class,自定义 ClassLoader 代码如下:

package com.wlee.test;

import org.apache.commons.io.IOUtils;
import org.apache.commons.lang3.reflect.FieldUtils;

import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.lang.reflect.Field;

public class MyClassLoader extends ClassLoader {

    private String path = "d:/test/";

    @Override
    protected Class<?> findClass(String name) throws ClassNotFoundException {
        Class<?> cls = findLoadedClass(name);
        if (cls != null) {
            return cls;
        }
        if (!name.endsWith(".App")) {
            return super.loadClass(name);
        }
        try {
            InputStream is = new FileInputStream(path + name.replace(".", "/") + ".class");
            byte[] bytes = IOUtils.toByteArray(is);
            return defineClass(name, bytes, 0, bytes.length);
        } catch (IOException e) {
            e.printStackTrace();
        }
        return super.loadClass(name);
    }

    public static void main(String[] args) {
        try {
            MyClassLoader loader = new MyClassLoader();
            Class<?> cls = loader.loadClass("com.wlee.test.App");
            Field field = FieldUtils.getField(cls, "name", true);
            Object value = field.get(cls.newInstance());
            System.out.println(value);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

执行main方法之后,自定义类加载器正常加载到类,程序最后输出:测试App。

代码很简单,上面那个 main 方法,就是读取自定义的目录。其实可以直接使用 URLClassLoader 就能读取:

    public static void main(String[] args) {
        try {
            URLClassLoader loader = new URLClassLoader(new URL[]{new URL("file:///d:/test/")});
            Class<?> cls = loader.loadClass("com.wlee.test.App");
            Field field = FieldUtils.getField(cls, "name", true);
            Object value = field.get(cls.newInstance());
            System.out.println(value);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

WorkLee

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值