广告位联系
返回顶部
分享到

JVM的类加载器和双亲委派模式你了解吗

java 来源:互联网 作者:秩名 发布时间:2022-03-13 19:46:29 人浏览
摘要

类加载器 Java虚拟机设计团队有意把类加载阶段中的通过一个类的全限定名来获取描述该类的二进制字节流这个动作放到Java虚拟机外部去实现,以便让应用程序自己决定如何去获取所需

类加载器

Java虚拟机设计团队有意把类加载阶段中的“通过一个类的全限定名来获取描述该类的二进制字节流”这个动作放到Java虚拟机外部去实现,以便让应用程序自己决定如何去获取所需的类。实现这个动作的代码被称为“类加载器”(ClassLoader)。

对于任意一个类,都必须由加载它的类加载器和这个类本身一起共同确立其在Java虚拟机中的唯一性,每一个类加载器,都拥有一个独立的类名称空间。这句话可以表达得更通俗一些:比较两个类是否“相等”,只有在这两个类是由同一个类加载器加载的前提下才有意义,否则,即使这两个类来源于同一个Class文件,被同一个Java虚拟机加载,只要加载它们的类加载器不同,那这两个类就必定不相等。

名称 加载的类 说明
Bootstrap ClassLoader(启动类加载器) JAVA_HOME/jre/lib 无法直接访问
Extension ClassLoader(拓展类加载器) JAVA_HOME/jre/lib/ext 上级为Bootstrap,显示为null
Application ClassLoader(应用程序类加载器) classpath 上级为Extension
自定义类加载器 自定义 上级为Application

1、启动类加载器

可通过在控制台输入指令,使得类被启动类加器加载,它是用C++写的,看不到源码;其他类加载器是用Java写的,说白了就是一些Java类,比如扩展类加载器、应用类加载器。

1

2

3

4

5

//查询所有被启动类加载器加载的类

URL[] urls = sun.misc.Launcher.getBootstrapClassPath().getURLs();

for (URL url : urls) {

    System.out.println(url);

}

1

2

3

4

5

6

7

8

9

//查询到的结果

file:/C:/Program%20Files/Java/jre1.8.0_131/lib/resources.jar

file:/C:/Program%20Files/Java/jre1.8.0_131/lib/rt.jar

file:/C:/Program%20Files/Java/jre1.8.0_131/lib/sunrsasign.jar

file:/C:/Program%20Files/Java/jre1.8.0_131/lib/jsse.jar

file:/C:/Program%20Files/Java/jre1.8.0_131/lib/jce.jar

file:/C:/Program%20Files/Java/jre1.8.0_131/lib/charsets.jar

file:/C:/Program%20Files/Java/jre1.8.0_131/lib/jfr.jar

file:/C:/Program%20Files/Java/jre1.8.0_131/classes

由上可以看出启动类加载的都是jre和jre/lib目录下的核心库,具体路径要看你的jre安装在哪里。

2、拓展类加载器

如果classpath和JAVA_HOME/jre/lib/ext 下有同名类,加载时会使用拓展类加载器加载。当应用程序类加载器发现拓展类加载器已将该同名类加载过了,则不会再次加载

1

2

3

4

URL[] urls = ((URLClassLoader) ClassLoader.getSystemClassLoader().getParent()).getURLs();

for (URL url : urls) {

    System.out.println(url);

}

1

2

3

4

5

6

7

8

9

10

11

12

13

file:/C:/Program%20Files/Java/jre1.8.0_131/lib/ext/access-bridge-64.jar

file:/C:/Program%20Files/Java/jre1.8.0_131/lib/ext/cldrdata.jar

file:/C:/Program%20Files/Java/jre1.8.0_131/lib/ext/dnsns.jar

file:/C:/Program%20Files/Java/jre1.8.0_131/lib/ext/dns_sd.jar

file:/C:/Program%20Files/Java/jre1.8.0_131/lib/ext/jaccess.jar

file:/C:/Program%20Files/Java/jre1.8.0_131/lib/ext/jfxrt.jar

file:/C:/Program%20Files/Java/jre1.8.0_131/lib/ext/localedata.jar

file:/C:/Program%20Files/Java/jre1.8.0_131/lib/ext/nashorn.jar

file:/C:/Program%20Files/Java/jre1.8.0_131/lib/ext/sunec.jar

file:/C:/Program%20Files/Java/jre1.8.0_131/lib/ext/sunjce_provider.jar

file:/C:/Program%20Files/Java/jre1.8.0_131/lib/ext/sunmscapi.jar

file:/C:/Program%20Files/Java/jre1.8.0_131/lib/ext/sunpkcs11.jar

file:/C:/Program%20Files/Java/jre1.8.0_131/lib/ext/zipfs.jar

这些类库具体是什么不重要,只需要知道不同的类库可能是被不同的类加载器加载的。

3、应用程序类加载器

1

2

3

4

URL[] urls = ((URLClassLoader) ClassLoader.getSystemClassLoader()).getURLs();

for (URL url : urls) {

    System.out.println(url);

}

1

file:/{项目工程目录}/bin/

这是当前java工程的bin目录,也就是我们自己的Java代码编译成的class文件所在。

4、双亲委派模式

双亲委派模式,调用类加载器ClassLoader 的 loadClass 方法时,查找类的规则。

loadClass源码

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

protected Class<?> loadClass(String name, boolean resolve)

    throws ClassNotFoundException

{

    synchronized (getClassLoadingLock(name)) {

        // 首先查找该类是否已经被该类加载器加载过了

        Class<?> c = findLoadedClass(name);

        //如果没有被加载过

        if (c == null) {

            long t0 = System.nanoTime();

            try {

                //看是否被它的上级加载器加载过了 Extension的上级是Bootstarp,但它显示为null

                if (parent != null) {

                    c = parent.loadClass(name, false);

                } else {

                    //看是否被启动类加载器加载过

                    c = findBootstrapClassOrNull(name);

                }

            } catch (ClassNotFoundException e) {

                // ClassNotFoundException thrown if class not found

                // from the non-null parent class loader

                //捕获异常,但不做任何处理

            }

            if (c == null) {

                //如果还是没有找到,先让拓展类加载器调用findClass方法去找到该类,如果还是没找到,就抛出异常

                //然后让应用类加载器去找classpath下找该类

                long t1 = System.nanoTime();

                c = findClass(name);

                // 记录时间

                sun.misc.PerfCounter.getParentDelegationTime().addTime(t1 - t0);

                sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(t1);

                sun.misc.PerfCounter.getFindClasses().increment();

            }

        }

        if (resolve) {

            resolveClass(c);

        }

        return c;

    }

}

有一个描述类加载器加载类过程的术语:双亲委派模型。然而这是一个很有误导性的术语,它应该叫做单亲委派模型(Parent-Delegation Model)。但是没有办法,大家都已经这样叫了。所谓双亲委派,这个亲就是指ClassLoader里的全局变量parent,也就是父加载器。

双亲委派的具体过程如下:

  • 当一个类加载器接收到类加载任务时,先查缓存里有没有,如果没有,将任务委托给它的父加载器去执行。
  • 父加载器也做同样的事情,一层一层往上委托,直到最顶层的启动类加载器为止。
  • 如果启动类加载器没有找到所需加载的类,便将此加载任务退回给下一级类加载器去执行,而下一级的类加载器也做同样的事情。
  • 如果最底层类加载器仍然没有找到所需要的class文件,则抛出异常。
  • 所以是一条线传上再传下,并没有什么“双亲”。

在这里插入图片描述

为什么要双亲委派?

答:确保类的全局唯一性。

如果你自己写的一个类与核心类库中的类重名,会发现这个类可以被正常编译,但永远无法被加载运行。因为你写的这个类不会被应用类加载器加载,而是被委托到顶层,被启动类加载器在核心类库中找到了。如果没有双亲委托机制来确保类的全局唯一性,谁都可以编写一个java.lang.Object类放在classpath下,那应用程序就乱套了。

从安全的角度讲,通过双亲委托机制,Java虚拟机总是先从最可信的Java核心API查找类型,可以防止不可信的类假扮被信任的类对系统造成危害。

5、自定义类加载器

如果我们自己去实现一个类加载器,基本上就是继承ClassLoader之后重写findClass方法,且在此方法的最后调包defineClass。

5.1、使用场景

  • 想加载非 classpath 随意路径中的类文件
  • 通过接口来使用实现,希望解耦时,常用在框架设计
  • 这些类希望予以隔离,不同应用的同名类都可以加载,不冲突,常见于 tomcat 容器

5.2、步骤

  • 继承ClassLoader父类
  • 要遵从双亲委派机制,重写 ?ndClass 方法
    • 不是重写loadClass方法,否则不会走双亲委派机制
  • 读取类文件的字节码
  • 调用父类的 de?neClass 方法来加载类
  • 使用者调用该类加载器的 loadClass 方法

1

2

3

4

5

6

7

protected Class<?> findClass(final String name) throws ClassNotFoundException {

    // 1、安全检查

    // 2、根据绝对路径把硬盘上class文件读入内存

    byte[] raw = getBytes(name);

    // 3、将二进制数据转换成class对象

    return defineClass(raw);

}


版权声明 : 本文内容来源于互联网或用户自行发布贡献,该文观点仅代表原作者本人。本站仅提供信息存储空间服务和不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权, 违法违规的内容, 请发送邮件至2530232025#qq.cn(#换@)举报,一经查实,本站将立刻删除。
原文链接 : https://blog.csdn.net/weixin_45379261/article/details/123446935
相关文章
  • SpringBoot自定义错误处理逻辑介绍

    SpringBoot自定义错误处理逻辑介绍
    1. 自定义错误页面 将自定义错误页面放在 templates 的 error 文件夹下,SpringBoot 精确匹配错误信息,使用 4xx.html 或者 5xx.html 页面可以打印错误
  • Java实现手写一个线程池的代码

    Java实现手写一个线程池的代码
    线程池技术想必大家都不陌生把,相信在平时的工作中没有少用,而且这也是面试频率非常高的一个知识点,那么大家知道它的实现原理和
  • Java实现断点续传功能的代码

    Java实现断点续传功能的代码
    题目实现:网络资源的断点续传功能。 二、解题思路 获取要下载的资源网址 显示网络资源的大小 上次读取到的字节位置以及未读取的字节
  • 你可知HashMap为什么是线程不安全的
    HashMap 的线程不安全 HashMap 的线程不安全主要体现在下面两个方面 在 jdk 1.7 中,当并发执行扩容操作时会造成环形链和数据丢失的情况 在
  • ArrayList的动态扩容机制的介绍

    ArrayList的动态扩容机制的介绍
    对于 ArrayList 的动态扩容机制想必大家都听说过,之前的文章中也谈到过,不过由于时间久远,早已忘却。 所以利用这篇文章做做笔记,加
  • JVM基础之字节码的增强技术介绍

    JVM基础之字节码的增强技术介绍
    字节码增强技术 在上文中,着重介绍了字节码的结构,这为我们了解字节码增强技术的实现打下了基础。字节码增强技术就是一类对现有字
  • Java中的字节码增强技术

    Java中的字节码增强技术
    1.字节码增强技术 字节码增强技术就是一类对现有字节码进行修改或者动态生成全新字节码文件的技术。 参考地址 2.常见技术 技术分类 类
  • Redis BloomFilter布隆过滤器原理与实现

    Redis BloomFilter布隆过滤器原理与实现
    Bloom Filter 概念 布隆过滤器(英语:Bloom Filter)是1970年由一个叫布隆的小伙子提出的。它实际上是一个很长的二进制向量和一系列随机映射
  • Java C++算法题解leetcode801使序列递增的最小交换次

    Java C++算法题解leetcode801使序列递增的最小交换次
    题目要求 思路:状态机DP 实现一:状态机 Java 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 class Solution { public int minSwap(int[] nums1, int[] nums2) { int n
  • Mybatis结果集映射与生命周期介绍

    Mybatis结果集映射与生命周期介绍
    一、ResultMap结果集映射 1、设计思想 对简单的语句做到零配置,对于复杂一点的语句,只需要描述语句之间的关系就行了 2、resultMap的应用场
  • 本站所有内容来源于互联网或用户自行发布,本站仅提供信息存储空间服务,不拥有版权,不承担法律责任。如有侵犯您的权益,请您联系站长处理!
  • Copyright © 2017-2022 F11.CN All Rights Reserved. F11站长开发者网 版权所有 | 苏ICP备2022031554号-1 | 51LA统计