前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >浅谈Java类加载器

浅谈Java类加载器

作者头像
Java学习录
发布2019-05-15 13:53:04
3060
发布2019-05-15 13:53:04
举报
文章被收录于专栏:Java学习录Java学习录Java学习录

JDK 默认提供了3种ClassLoader:

  • Bootstrp Classloader(启动类加载器) 由C++编写,它主要负责加载Java的核心类库:
    • %JAVA_HOME%/jre/lib
    • -Xbootclasspath参数指定的路径
    • %JAVA_HOME%/jre/classes中的类

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

  • Extension ClassLoader(扩展类加载器) 加载加载扩展类,扩展JVM的类库它主要负责加载以下类库:
    • %JAVA_HOME%/jre/lib/ext目录
    • java.ext.dirs系统变量指定的路径中类库
  • Application ClassLoader (应用程序加载器) Java程序默认的类加载器,主要负责加载classpath所指定的类库

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

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

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


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.....
xtClassLoader....
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;
    }
}
本文参与 腾讯云自媒体分享计划,分享自微信公众号。
原始发表:2019-05-10,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 Java学习录 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档