浅谈Java类加载器

本文深入探讨了Java虚拟机的类加载机制,包括三种默认的类加载器:BootstrapClassLoader、ExtensionClassLoader和ApplicationClassLoader,以及它们之间的层次关系和双亲委派模型。详细解析了类加载的过程和loadClass()方法的实现。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

JDK 默认提供了3种ClassLoader:

  • Bootstrp Classloader(启动类加载器)

    • %JAVA_HOME%/jre/lib

    • -Xbootclasspath参数指定的路径

    • %JAVA_HOME%/jre/classes中的类

需要注意的是这个加载器是虚拟机启动时初始化的,Java程序是无法使用的

  • Extension ClassLoader(扩展类加载器)

    • %JAVA_HOME%/jre/lib/ext目录

    • java.ext.dirs系统变量指定的路径中类库

  • Application ClassLoader (应用程序加载器)

除了以上3个类加载器,开发人员还可以根据自己的需求自己定制实现类加载器

类加载器之间的层次模型一般是这样的:

640?wx_fmt=png

上图展示的类加载器之间的这种层次关系,使用如下代码可证明:

public class Test {	
    public static void main(String args[])	
        System.out.println(ClassLoader.getSystemClassLoader());	
        System.out.println(ClassLoader.getSystemClassLoader().getParent());	
        System.out.println(ClassLoader.getSystemClassLoader().getParent().getParent());	
    }	
}

上述代码的依次打印结果为:

AppClassLoader....	
ExtClassLoader....	
Null

注意最后的null值应该是启动类加载器、单因为是C++语言编写的,所以无法获取到相关的信息

上述的这种层级关系被称为类加载器的双亲委派模型。此模型要求除了最顶层的启动类加载器外,其余的加载都应该有自己的父加载器。

当一个类加载器在接到加载类的请求时,首先将加载任务委托交给父类加载器,父类加载器又将加载任务向上委托,直到最顶层的启动类加载器,如果父类加载器可以完成类加载任务,就成功返回,如果不行就向下传递委托任务,由其子类加载器进行加载。

其过程可见java.lang.ClassLoader的loadClass()方法

protected Class<?> loadClass(String name, boolean resolve)	
    throws ClassNotFoundException	
{	
    synchronized (getClassLoadingLock(name)) {	
        // 检查class是否被加载过	
        Class<?> c = findLoadedClass(name);	
        if (c == null) {	
            long t0 = System.nanoTime();	
            try {	
                //如果父类不为空则委托父类加载	
                if (parent != null) {	
                    c = parent.loadClass(name, false);	
                } else {	
                //父类为空请求启动类加载器加载	
                    c = findBootstrapClassOrNull(name);	
                }	
            } catch (ClassNotFoundException e) {	
                // 如果父类抛出异常代表父类无法加载	
            }	
            //父类没有加载	
            if (c == null) {	
                // 使用本身的findClass方法加载	
                long t1 = System.nanoTime();	
                c = findClass(name);	

	
                // this is the defining class loader; record the stats	
                sun.misc.PerfCounter.getParentDelegationTime().addTime(t1 - t0);	
                sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(t1);	
                sun.misc.PerfCounter.getFindClasses().increment();	
            }	
        }	
        if (resolve) {	
            resolveClass(c);	
        }	
        return c;	
    }	
}

640?wx_fmt=jpeg

万水千山总是情,点个 “在看” 行不行!!!

640?wx_fmt=png 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值