最近在看面试题,发现很多面试题在介绍双亲委派机制时,都只是很粗糙的介绍:
“双亲委派机制是指当一个类加载器收到一个类加载请求时,该类加载器首先会把请求委派给父类加载器。每个类加载器都是如此,只有在父类加载器在自己的搜索范围内找不到指定类时,子类加载器才会尝试自己去加载。”
但唯独有一篇更详细的介绍提到当类加载器接收到加载请求后,会首先查看当前类加载器是否加载过该类,防止重复。如果没有重复才会委托双亲进行加载。
我们从源码的角度去看看到底是怎么样的?
protected Class<?> loadClass(String name, boolean resolve) throws ClassNotFoundException
{
synchronized (getClassLoadingLock(name)) {
// 查看该类是否被当前加载器加载
Class<?> c = findLoadedClass(name);
if (c == null) {
long t0 = System.nanoTime();
try {
// 如果父加载器存在,则调用父加载器的方法
if (parent != null) {
c = parent.loadClass(name, false);
} else {
// 父加载器 == null,说明是引导类加载器,查看引导类加载器是否加载
c = findBootstrapClassOrNull(name);
}
} catch (ClassNotFoundException e) {
// ClassNotFoundException thrown if class not found
// from the non-null parent class loader
}
// 当前类加载器的父加载器未加载国,且尝试加载,加载失败,则当前类加载器尝试加载
if (c == null) {
// If still not found, then invoke findClass in order
// to find the class.
long t1 = System.nanoTime();
// 核心,调用findClass,读取字节流,尝试加载类,定义类对象
c = findClass(name);
// this is the defining class loader; record the stats
sun.misc.PerfCounter.getParentDelegationTime().addTime(t1 - t0);
sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(t1);
sun.misc.PerfCounter.getFindClasses().increment();
}
}
if (resolve) {
resolveClass(c);
}
return c;
}
}
我们可以看到,它以一种类似递归的实现,简洁地实现了双亲委派模型:
1、当类加载器接收到加载请求后,会首先查看当前类加载器是否加载过该类,防止重复。
2、委托双亲进行加载
3、双亲没加载成功,则调用findClass,尝试加载类
评论区