Java类的双亲委派机制
java中类的双亲委派机制是工作面试中的高频面试问题,要想搞清楚什么是双亲委派,首先要明白类加载器之间的关系与工作方式
类加载器
类加载器(ClassLoader)是Java虚拟机提供给程序去获取类和接口字节码数据的技术,类加载器只参与加载过程中的字节码信息获取并加载到内存这一部分,类加载器分为两大类,一类是Java代码中实现的,另一种是Java虚拟机底层源码实现的。
所有的Java代码中实现的类加载器都继承自ClassLoader这个抽象类
而常用的类加载器共有三个:
- 启动类加载器Bootstrap:加载Java中最核心的类,默认加载Java安装目录/jre/lib下的类文件,比如rt.jar、tools.jar等,若在控制台输出由启动类加载器加载的类的
getClassLoader()
方法,返回值为null - 扩展类加载器Extension:扩展Java中比较通用的类,默认加载Java安装目录/jre/lib/ext下的类文件,
- 应用类加载器Application:加载应用使用的类,默认加载classpath下的类,比如自己编写的类和maven引入的依赖jar包
双亲委派
如果有一个类,在三个类加载器的默认加载目录下都存在,那么是由哪一个类加载器来加载呢?
其实三个类加载器是有层次结构的:
双亲委派就是当一个类加载器接收到加载类的任务时,会自底向上
查找是否被加载过,再由顶向下
进行加载。
简单来说,当有类加载任务时,类加载器会先检查自己是否以及加载过该类,如果已经加载则返回,若没有加载过则委派给父类加载器。若直到启动类加载器Bootstrap也没有加载过,会判断该类是否在默认加载目录中,若在则加载该类,若没有则给子类加载器加载。
好处与作用
类加载器采用这样的双亲委派机制有什么作用与好处呢
- 保证类加载的安全性
通过双亲委派机制避免恶意代码替换JDK中的核心类库,比如java.lang.String类,如果自己写一个java.lang.String类,在类加载时是无法替代的,在加载过程中,Bootstrap加载器已经加载过java.lang.String类,再次加载自己写的类时,向上查找时就会查找到该类已经被加载过,不会被再次加载 - 避免重复加载
双亲委派机制可以避免一个类被多个类加载器多次加载,提高加载效率与规范