为什么JVM使用双亲委派?

1.为什么要有双亲委派?

首先明确一点:jvm如何认定两个对象同属于一个类型,必须同时满足下面两个条件:

  • 都是用同名的类完成实例化的。
  • 两个实例各自对应的同名的类的加载器必须是同一个。比如两个相同名字的类,一个是用系统加载器加载的,一个扩展类加载器加载的,两个类生成的对象将被jvm认定为不同类型的对象。

所以,为了系统类的安全,类似java.lang.Object这种核心类,jvm需要保证他们生成的对象都会被认定为同一种类型
即:通过代理模式,对于 Java 核心库的类的加载工作由启动类加载器(bootstrap)来统一完成,保证了 Java 应用所使用的都是同一个版本的 Java 核心库的类,是互相兼容的

说白了就是,遇到一个类,都交给上层处理,这样在不同的地方加载同一个类比如Object时,两个地方都会把Object往上递交给启动类(引导类)加载器,那么生成的对象就是同一个类型(Object类型)的。越往上,覆盖的范围越大。

如果父加载器表示自己无法完成,才会让子类加载器加载。

比如有类加载请求,那么这个请求会被一层层往上

自定义加载器1 -> 应用程序类加载器->扩展类加载器->启动类加载器

一直传到了启动类加载器,

(1)启动类加载器说,我处理不了,儿子(扩展类加载器)你自己处理吧

(2)扩展类加载器一看,自己也处理不了,说 我也处理不了,儿子(应用程序类加载器)你自己处理吧

(3)应用程序类加载器一看,自己也处理不了,说 俺还是没办法处理,儿子(自定义类加载器1)你自己处理吧(一般我们自己写的Java类都在这一层加载)

(4)自定义类加载器1说,你们都不行,我一层一层的往上给你们传,让你们先处理,结果你们都处理不了,那还是我自己来处理把!

2.什么叫类加载器处理不了?

BootStrap启动类加载器为例子,

该加载器由C++实现,加载的是<JAVA_HOME>/lib下的class文件,或-Xbootclasspath参数指定的路径下的jar包加载到内存中,注意必由于虚拟机是按照文件名识别加载jar包的,如rt.jar,如果文件名不被虚拟机识别,即使把jar包丢到lib目录下也是没有作用的(出于安全考虑,Bootstrap启动类加载器只加载包名为java、javax、sun等开头的类)。所以,字节码class文件如果不是这种形式,比如我们瞎写的类,它就不可能被bootstrap加载。

扩展类加载器,由Java语言实现的,它负责加载<JAVA_HOME>/lib/ext目录下或者由系统变量-Djava.ext.dir指定位路径中的类库,开发者可以直接使用标准扩展类加载器。

系统类加载器也称应用程类加载器 它负责加载系统类路径java -classpath-D java.class.path指定路径下的类库,也就是我们经常用到的classpath路径,开发者可以直接使用系统类加载器,一般情况下该类加载是程序中默认的类加载器,通过ClassLoade.getSystemClassLoader()方法可以获取到该类加载器。

系统自带三个类加载器都加载特定目录下的类,如果我们自己的类加载器加载一个特殊的目录,那么系统的加载器就无法加载,也就是最终还是由我们自己的加载器加载。

3.能不能自己写个类叫java.lang.System?

答案:通常不可以,但可以采取另类方法达到这个需求。

解释:为了不让我们写System类,类加载采用委托机制,这样可以保证爸爸们优先,爸爸们能找到的类,儿子就没有机会加载。而System类是Bootstrap加载器加载的,就算自己重写,也总是使用Java系统提供的System,自己写的System类根本没有机会得到加载。

但是,我们可以自己定义一个类加载器来达到这个目的,为了避免双亲委托机制,这个类加载器也必须是特殊的。由于系统自带的三个类加载器都加载特定目录下的类,如果我们自己的类加载器加载一个特殊的目录,那么系统的加载器就无法加载,也就是最终还是由我们自己的加载器加载

  • 4
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 5
    评论
JVM中的双亲委派机制是一种加载机制,它规定了在Java中一个加载时如何进行类加载器的选择。根据这个机制,当一个需要被加载时,首先会由类加载器ClassLoader检查是否已经加载过该,如果是,则直接返回已经加载过的;如果不是,则将该请求委派给父类加载器加载。这样的过程会一直向上委派,直到达到顶层的引导类加载器(Bootstrap ClassLoader)。引用 引用中提到,并不是所有的类加载器都采用双亲委派机制。Java虚拟机规范并没有强制要求使用双亲委派机制,只是建议使用。实际上,一些类加载器可能会采用不同的加载顺序,例如Tomcat服务器类加载器就是采用代理模式,首先尝试自己去加载某个,如果找不到再代理给父类加载器。 引用中提到,引导类加载器(Bootstrap ClassLoader)是最早开始工作的类加载器,负责加载JVM的核心库,例如java.lang.*包中的。这些JVM启动时就已经被加载到内存中。 综上所述,JVM双亲委派机制是一种加载机制,它通过类加载器的委派方式来加载,首先检查是否已经加载过该,如果没有则委派给父类加载器加载,直到达到顶层的引导类加载器。不过,并不是所有的类加载器都采用该机制,一些类加载器可能会采用不同的加载顺序。引用<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [JVM-双亲委派机制](https://blog.csdn.net/m0_51608444/article/details/125835862)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] - *2* *3* [jvm-双亲委派机制](https://blog.csdn.net/y08144013/article/details/130724858)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值