【JVM】双亲委派模型

双亲委派模型(面试高频)

之所以这个东西面试出场概率高高,最大的原因就是它起的名字很好听


描述了查找 .class 文件的策略。

类加载器

JVM 中进行类加载的操作,是有一个专门的模块,称为“类加载器”(ClassLoader)。JVM 中的类加载器默认是由三个(也可以自定义个数)

类加载器的作用,给它一个“权限定类名”(带有包的类名(java. lang. String)),给了之后,就找到对应的 .class 文件。这里的类加载器就是从不同的目录中进行查找

  1. BootstrapClassLoader

    • 负责查找标准库的目录
  2. ExtensionClassLoader

    • 负责查找扩展库的目录
    • Java 语法的规范里面描述了标准库中应该有哪些功能。实现 JVM 的厂商/组织会在标准库的基础上扩充一些额外的功能(JVM 内置的,不同的厂商扩展的可能不太一样)
    • 这块内容在上古时期用处比较多,但随着时代的发展,这里的内容很少会使用了
  3. ApplicationClassLoader

    • 负责查找当前项目的代码目录,以及第三方库的目录

上述的三个类加载器,存在“父子关系image.png|249

  • 不是面向对象中的,父类子类继承关系
  • 而是类似于“二叉树”。有一个指针(引用)parent,指向自己的“父”类加载器

双亲委派模型,就描述了上述类加载器之间是符合配合工作的

工作过程

双亲委派模型工作过程:

  1. ApplicationClassLoader 作为入口,先开始工作

  2. ApplicationClassLoader 不会立即搜索自己负责的目录,会把搜索的任务交给自己的父亲

  3. 代码就进入到 ExtensionClassLoader 范畴了,但它也不会立即搜索自己负责的目录,也要把搜索的任务交给自己的父亲

  4. 代码就进入到 BootstrapClassLoader 范畴了,它也不想立即搜索自己负责的目录,也想把搜索的任务交给自己的父亲

  5. BooststracpClassLoader 发现自己没有父亲,才会真正搜索负责的目录(标准库目录),通过全限定类名,尝试在标准库目录中找到符合要求的 .class 文件

    • 如果找到了,接下来就直接进入到打开文件/读文件等流程中
    • 如果没找到,就回到孩子这一辈的类加载器中,继续尝试加载
  6. ExtensionClassLoader 收到父亲交回给他的任务之后,自己进行搜索负责目录(扩展库的目录)

    • 如果找到了,接下来就进入到后续流程
    • 如果没找到,也就再回到孩子这一辈的类加载器中继续尝试加载
  7. ApplicationClassLoader 收到父亲交回给他的任务之后,自己进行搜索负责的目录(当前项目目录/第三方库目录)

    • 如果找到了,接下来就进入后续流程
    • 如果没找到,也是回到孩子这一辈的类加载器中尝试继续加载。但是由于默认情况下 ApplicationClassLoader 没有孩子了,此时说明类加载过程失败了,就会抛出 ClassNotFoundException 异常

存在意义

上述设定的最主要的目的就是为了确保这几个类加载器之间的优先级

按照上述的顺序,假定在代码中自己定义了一个 java.lang.String 这样的类。最终程序执行效果是:自定义的类不会被 JVM 加载

  • 因为当我们拿着这个类给类加载器找的时候,先 BootstrapClassLoader 到标准库中找,就找到了这个类,直接就从标准库中加载了,就不会回到 ApplicationClassLoader (你自己写的代码中)这一层进行加载
    设定就是为了避免你写的类的名字和标准库的重复了,导致标准库的类的功能失效
### 回答1: 我不认为JVM可以“打破”双亲委派模型JVM使用双亲委派模型是为了保证类的加载和安全性,这是JVM的基本设计原则之一。 在双亲委派模型中,当一个类被加载时,首先会委托其父类加载器尝试加载该类。如果父类加载器无法加载该类,则会将该请求传递给其父类加载器的父类加载器,直到最终传递到启动类加载器。如果启动类加载器也无法加载该类,则会抛出ClassNotFoundException异常。 这种模型的好处是可以防止类的重复加载,并且可以保证类的安全性,因为只有在受信任的类加载器中加载的类才能访问受信任的资源。如果JVM打破了这种模型,可能会导致类的重复加载和不安全的类访问,从而破坏JVM的基本设计原则。 虽然JVM不能打破双亲委派模型,但是它提供了一些机制来绕过该模型。例如,可以使用Java的反射机制或者自定义类加载器来加载类,但是这些机制都需要开发者自己负责类的加载和安全性,因此需要谨慎使用。 ### 回答2: JVM双亲委派模型是一种类加载机制,用于保证Java程序的安全性和稳定性。在双亲委派模型中,JVM首先会查找并加载自定义类加载器的类,如果找不到,才会向上一级的父类加载器请求加载。这种模型的好处是可以避免类重复加载和类冲突,确保程序的运行稳定性。 但有时候,我们可能需要打破双亲委派模型,以满足特定的需求。下面是一些打破双亲委派模型的方式: 1. 自定义类加载器:可以自己编写一个类加载器,并重写其中的加载类的逻辑。这样就可以绕过父类加载器,直接加载自定义的类。 2. 加载本地库:JVM的类加载器无法加载本地库,因此可以通过JNI(Java Native Interface)来打破双亲委派模型,直接加载本地库。 3. 使用反射技术:通过反射,可以调用私有的加载类方法,从而绕过双亲委派模型,加载特定的类。 4. 使用SPI机制:在Java标准库中,有一些接口通过SPI(Service Provider Interface)机制提供了灵活的类加载方式,可以通过实现自己的SPI来打破双亲委派模型。 需要注意的是,打破双亲委派模型可能会导致类的重复加载和类冲突,引发程序的运行问题。因此,在使用这些方法时,需要谨慎考虑,确保安全性和稳定性。
评论 64
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值