带你看懂双亲委派,耗子尾汁
双亲委派机制
双亲委派机制是指当一个类加载器收到一个类加载请求时,该类加载器首先会把请求委派给父类加载器。每个类加载器都是如此(递归的去查找),只有在父类加载器在自己的搜索范围内找不到指定类时,子类加载器才会尝试自己去加载。
显然,在介绍双亲委派机制的时候,不得不提ClassLoader。再说ClassLoader之前,我们得先了解下Java的基本知识。
Java是运行在Java的虚拟机(JVM)中的,但是它是怎么就运行在JVM中了呢?我们在IDE中编写的Java源代码被编译器编译成.class
的字节码文件。然后由我们的ClassLoader
负责将这些class文件加载到JVM
中去执行。
JVM中提供了三层的ClassLoader:
大家好,我是AIO生活,关注我,后续连载更多技术重难点,文章有不足之处,欢迎大家留言指正,谢谢大家啦!
顺便提一下,帮大家整理了些JAVA电子书,扫码回复“1024”获取电子书合集,也可签到领现金红包。
Bootstrap ClassLoader(启动类加载器):主要负责加载核心的类库(java.lang.*等),构造Extension ClassLoader
和Application ClassLoader
。
Extension ClassLoader(扩展类加载器):主要负责加载jre/lib/ext
目录下的一些扩展的jar。
Application ClassLoader(应用程序类加载器):主要负责加载应用程序
的主函数类。
最后一个CustomClassLoader(用户自定义类加载器) 是java
编写,用户自定义的类加载器,可加载指定路径的class文件
。
那如果有一个Hello.class
文件是如何被加载到JVM中的呢?
我们简单看一下源码
protected Class<?> loadClass(String name, boolean resolve) throws ClassNotFoundException { synchronized (getClassLoadingLock(name)) { // 首先检查这个classsh是否已经加载过了 Class<?> c = findLoadedClass(name); if (c == null) { long t0 = System.nanoTime(); try { // c==null表示没有加载,如果有父类的加载器则让父类加载器加载 if (parent != null) { c = parent.loadClass(name, false); } else { //如果父类的加载器为空 则说明递归到bootStrapClassloader了 //bootStrapClassloader比较特殊无法通过get获取 c = findBootstrapClassOrNull(name); } } catch (ClassNotFoundException e) {} if (c == null) { //如果bootstrapClassLoader 仍然没有加载过,则递归回来,尝试自己去加载class 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; } }
上述这段代码已经很好的解释了双亲委派机制,为了更容易理解,我们先参看下面描述上述代码流程的图再谈Hello.class
如何加载:
从上图中我们就更容易理解了,当一个Hello.class
这样的文件要被加载时。不考虑我们自定义类加载器,首先会在AppClassLoader中检查是否加载过,如果有那就无需再加载了。如果没有,那么会拿到父加载器,然后调用父加载器的loadClass方法。父类中同理会先检查自己是否已经加载过,如果没有再往上。注意这个过程,直到到达Bootstrap ClassLoader之前,都是没有哪个加载器自己选择加载的。如果父加载器无法加载,会下沉到子加载器去加载,一直到最底层(其实就是递归查找过程),如果没有任何加载器能加载,就会抛出ClassNotFoundException
。
于是,我们就可以很好的总结双亲委派机制的工作流程了:
- 1、当
Application ClassLoader
收到一个类加载请求时,他首先不会自己去尝试加载这个类,而是将这个请求委派给父类加载器Extension ClassLoader
去完成。 - 2、当
Extension ClassLoader
收到一个类加载请求时,他首先也不会自己去尝试加载这个类,而是将请求委派给父类加载器Bootstrap ClassLoader
去完成。 - 3、如果
Bootstrap ClassLoader
加载失败(在<JAVA_HOME>\lib中未找到所需类),就会让Extension ClassLoader
尝试加载。 - 4、如果
Extension ClassLoader
也加载失败,就会使用Application ClassLoader
加载。 - 5、如果
Application ClassLoader
也加载失败,就会使用Custom ClassLoader
(用户自定义加载器)去尝试加载。 - 6、如果均加载失败,就会抛出
ClassNotFoundException
异常。
双亲委派机制的作用
1、防止重复加载同一个.class
。通过委托去向上面问一问,加载过了,就不用再加载一遍。保证数据安全。
2、保证核心.class
不能被篡改。通过委托方式,不会去篡改核心.class
,即使篡改也不会去加载,即使加载也不会是同一个.class
对象了。不同的加载器加载同一个.class
也不是同一个Class对象。这样保证了Class执行安全。
举个栗子:如果有人想替换系统级别的类:String.java
。篡改它的实现,但是在这种机制下这些系统的类已经被Bootstrap ClassLoader
加载过了,所以并不会再去加载,从一定程度上防止了危险代码的植入。
大家好,我是AIO生活,关注我,后续连载更多技术重难点,文章有不足之处,欢迎大家留言指正,谢谢大家啦!
顺便提一下,帮大家整理了些JAVA电子书,扫码回复“1024”获取电子书合集,也可签到领现金红包。

低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。
持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。
- 上一篇
go-zero 如何扛住流量冲击(二)
本篇文章承接上一篇go-zero 如何扛住流量冲击(一)。 上一篇介绍的是 go-zero 中滑动窗口限流,本篇介绍另外一个 tokenlimit ,令牌桶限流。 使用 const ( burst = 100 rate = 100 seconds = 5 ) store := redis.NewRedis("localhost:6379", "node", "") fmt.Println(store.Ping()) // New tokenLimiter limiter := limit.NewTokenLimiter(rate, burst, store, "rate-test") timer := time.NewTimer(time.Second * seconds) quit := make(chan struct{}) defer timer.Stop() go func() { <-timer.C close(quit) }() var allowed, denied int32 var wait sync.WaitGroup for i := 0; i &a...
- 下一篇
申通快递 双11 云原生应用实践
来源 | 阿里巴巴云原生公众号 作者 | 溪恒、遥方 一年一度的 “双11” 大促中,交易额每年都在刷新,承接这些交易商品的快递包裹的数量也在成倍增长。这些快速的增长对物流系统带来了巨大的挑战,让物流管理更加敏捷来应对 “双11” 成为了必须解决的问题。 申通是目前国内最大的物流公司之一,为了解决 “双11” 的技术挑战,申通在物流场景引入 IOT、大数据和 AI 等先进和创建的技术手段,通过不断的技术快速迭代,使得物流资源得到有效的配置,推动了物流行业的发展。 快速的技术迭代带来了对 IT 基础设施的挑战,申通近年来全面将传统应用切换使用云原生架构,通过云原生架构实现应用的快速迭代、稳定的高可用、资源的动态扩缩容来支撑起快速的技术创新。 上云前,申通使用线下机房作为计算及数据存储平台,一到 双11 资源需求就膨胀,大促之后则闲置浪费;上云和云原生化后,几乎全部的资源都是按量购买,使用云原生技术快速扩缩容,双11 前快速扩容,双11 释放,真正做到了开箱即用,不产生一天浪费。与去年 双11 当天相比,今年 11 月 1 到 3 日,在业务量大幅提升的情况下,IT 投入反而降低了 30%...
相关文章
文章评论
共有0条评论来说两句吧...
文章二维码
点击排行
推荐阅读
最新文章
- Docker快速安装Oracle11G,搭建oracle11g学习环境
- SpringBoot2初体验,简单认识spring boot2并且搭建基础工程
- CentOS7编译安装Gcc9.2.0,解决mysql等软件编译问题
- Docker安装Oracle12C,快速搭建Oracle学习环境
- SpringBoot2全家桶,快速入门学习开发网站教程
- SpringBoot2配置默认Tomcat设置,开启更多高级功能
- CentOS关闭SELinux安全模块
- Eclipse初始化配置,告别卡顿、闪退、编译时间过长
- CentOS6,CentOS7官方镜像安装Oracle11G
- SpringBoot2整合MyBatis,连接MySql数据库做增删改查操作