类加载之双亲委派

双亲委派(Parent Delegation)是 Java 类加载机制中的一个重要概念。它指的是当一个类加载器(ClassLoader)需要加载某个类时,它会首先将加载请求委派给它的父类加载器,依次递归,直到顶层的启动类加载器(Bootstrap ClassLoader)。只有在父类加载器无法加载该类时,才会由子类加载器尝试加载。

这种机制的优势在于确保了类的唯一性避免了同一个类被多个类加载器加载导致的类重复问题。当类加载器遵循双亲委派模型时,类加载器之间形成了一种层级关系,底层的加载器可以利用上层加载器已经加载过的类,从而实现了类的共享与复用

这种机制也有助于保护核心 Java 类库,因为核心类库通常由启动类加载器加载,而用户自定义的类则由底层的类加载器加载,这样可以防止用户自定义的类覆盖核心类库中的类。先会加载核心类库中的类

总的来说,双亲委派机制有助于保证类的唯一性、避免类的重复加载,同时提高了类加载器的安全性和稳定性。


Java是运行在Java的虚拟机(JVM)中的,但是它是如何运行在JVM中了呢?我们在IDE中编写的Java源代码被编译器编译成.class的字节码文件。然后由我们的ClassLoader负责将这些class文件给加载到JVM中去执行。

JVM中提供了三层的ClassLoader:

Bootstrap classLoader:主要负责加载核心的类库(java.lang.*等),构造ExtClassLoader和APPClassLoader。
![[1460000043379721.webp]]

ExtClassLoader:主要负责加载jre/lib/ext目录下的一些扩展的jar。
![[1460000043379723.webp]]

AppClassLoader:主要负责加载当前classpath下面的所有jar包

那如果有一个我们写的Hello.java编译成的Hello.class文件,它是如何被加载到JVM中的呢?别着急,请继续往下看。

![[Pasted image 20240419190221.png]]

其实在双亲委派之前会有一个缓存的判断:
当加载一个类时,会先从应用程序类加载器的缓存里面找到相应的类,如果找到就返回对象,如果找不到就执行下面流程

![[Pasted image 20240419190315.png]]

从上图中我们就更容易理解了,当一个Hello.class这样的文件要被加载时。不考虑我们自定义类加载器,首先会在AppClassLoader检查是否加载过,如果有那就无需再加载了。如果没有,那么会拿到父加载器,然后调用父加载器的loadClass方法。父类中同理也会先检查自己是否已经加载过,如果没有再往上。注意这个类似递归的过程,直到到达Bootstrap classLoader之前,都是在检查是否加载过,并不会选择自己去加载。直到BootstrapClassLoader,已经没有父加载器了,这时候开始考虑自己是否能加载了,如果自己无法加载,会下沉到子加载器去加载,一直到最底层,如果没有任何加载器能加载,就会抛出ClassNotFoundException。那么有人就有下面这种疑问了?

关于为什么一个 Java 文件在 Bootstrap ClassLoader 中可以加载,可能是因为该 Java 文件中使用了 Java 核心类库中的一些类,比如 java.lang.String。由于 Bootstrap ClassLoader 负责加载核心类库,因此在加载 Java 文件时会由 Bootstrap ClassLoader 加载这些核心类库中的类。
一个java文件中使用的不同类由不同的类加载器去加载,最后形成一个类实例,在JVM中存储

为什么要设计这种机制


这种设计有个好处是,如果有人想替换系统级别的类:String.java。篡改它的实现,在这种机制下这些系统的类已经被Bootstrap classLoader加载过了(为什么?因为当一个类需要加载的时候,最先去尝试加载的就是BootstrapClassLoader),所以其他类加载器并没有机会再去加载,从一定程度上防止了危险代码的植入。
![[Pasted image 20240419190722.png]]

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值