侧边栏壁纸
博主头像
小吴同学博主等级

去学习,去读书,去受教育,去看山川河流,去远方!!!

  • 累计撰写 15 篇文章
  • 累计创建 9 个标签
  • 累计收到 2 条评论

目 录CONTENT

文章目录

从源码的角度理解类加载器双亲委派机制

小吴同学
2022-10-15 / 0 评论 / 14 点赞 / 86 阅读 / 560 字

最近在看面试题,发现很多面试题在介绍双亲委派机制时,都只是很粗糙的介绍:

“双亲委派机制是指当一个类加载器收到一个类加载请求时,该类加载器首先会把请求委派给父类加载器。每个类加载器都是如此,只有在父类加载器在自己的搜索范围内找不到指定类时,子类加载器才会尝试自己去加载。”

但唯独有一篇更详细的介绍提到当类加载器接收到加载请求后,会首先查看当前类加载器是否加载过该类,防止重复。如果没有重复才会委托双亲进行加载。
我们从源码的角度去看看到底是怎么样的?

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,尝试加载类

14

评论区