前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >JVM系列十三(类加载器).

JVM系列十三(类加载器).

作者头像
JMCui
发布2020-04-17 15:16:57
4700
发布2020-04-17 15:16:57
举报
文章被收录于专栏:JMCuiJMCui

一、概述

类加载器通过一个类的全限定名来获取描述此类的二进制字节流。

类加载器在类层次划分、OSGi、热部署、代码加密等领域发挥着重要的作用。

比较两个类是否“相等”,只有在这两个类是由同一个类加载器加载的前提之下才有意义,否则,即使这两个类是来源于同一个 Class 文件,只要加载它们的类加载器不同,那这两个类就必定不相等。这里的“相等”包括 equal() 方法、isAssignableForm() 方法、isInstance() 方法和 instanceof 关键字。

下面的例子可以看到,虽然都是来自同一个 Class 文件,但是因为类加载器不同,依然是两个独立的类,自然不会“相等”。

public static void main(String[] args) throws ClassNotFoundException, IllegalAccessException, InstantiationException {
    // 自定义简单类加载器
    ClassLoader myClassLoader = new ClassLoader() {
        @Override
        public Class<?> loadClass(String name) throws ClassNotFoundException {
            try {
                String fileName = name.substring(name.lastIndexOf(".") + 1) + ".class";
                InputStream inputStream = getClass().getResourceAsStream(fileName);
                if (inputStream == null) {
                    return super.loadClass(name);
                }
                byte[] bytes = new byte[inputStream.available()];
                inputStream.read(bytes);
                return defineClass(name, bytes, 0, bytes.length);
            } catch (IOException e) {
                e.printStackTrace();
            }
            return super.loadClass(name);
        }
    };
    Object newInstance = myClassLoader.loadClass("org.jvm.demo.chapter7.ClassLoaderTest").newInstance();
    System.out.println(newInstance.getClass()); // org.jvm.demo.chapter7.ClassLoaderTest
    System.out.println(newInstance instanceof org.jvm.demo.chapter7.ClassLoaderTest); // false
}

绝大部分 Java 程序都会使用到以下三种系统提供的类加载器:

  • 启动类加载器(Bootstrap ClassLoader):负责加载 JAVA_HOME\lib 或着 -Xbootclasspath 参数指定目录下的类库,加载内容按文件名识别,如 rt.jar,启动类加载器无法被 Java 程序直接引用。
  • 扩展类加载器(Extension ClassLoader):负责加载 JAVA_HOME\lib\ext 或者 java.ext.dirs 系统变量所指定的所有类库,该加载器由 sun.misc.Launcher$ExtClassLoader 实现,开发者可以直接使用扩展类加载器 — Launcher.getLauncher().getClassLoader()。
  • 应用程序类加载器(Application ClassLoader):负责加载用户类路径 ClassPath 上所指定的类库,如果应用程序没有自定义过自己的类加载器,一般情况下就是程序的默认类加载器,该加载器由 sun.misc.Launcher$AppClassLoader 实现,开发者可以直接使用这个类加载器 — ClassLoader.getSystemClassLoader()。

二、双亲委派模型

如图所示类加载器之间的层次关系,就称为类加载器的双亲委派模型。双亲委派模型要求除了顶层的启动类加载器外,其余的类加载器都应当有自己的父类加载器。这里类加载器之间的父子关系一般不会以继承的关系来实现,而是都使用组合关系来复用父加载器的代码。

双亲委派模型的工作过程是:如果一个类加载器收到了类加载的请求,它首先不会自己去尝试加载这个类,而是把这个请求委派给父类加载器去完成,每一个层次的类加载器都是如此。

双亲委派模型对于保证 Java 程序的稳定运作很重要,它让 Java 类随着它的类加载器一起具备了一种带有优先级的层次关系。

双亲委派模型不是一个强制性的约束模型,而是 Java 设计者们推荐给开发者们的一种类加载器的实现方式。

本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
原始发表:2020-04-16 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

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

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

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 一、概述
  • 二、双亲委派模型
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档