Tomcat打破双亲委派模型
阿里云国内75折 回扣 微信号:monov8 |
阿里云国际,腾讯云国际,低至75折。AWS 93折 免费开户实名账号 代冲值 优惠多多 微信号:monov8 飞机:@monov6 |
tomcat的类加载器结构
tomcat的类加载loadClass过程
和原本的双亲委派模型思路差不多先看有没有加载过。
先在本地 Cache 查找该类是否已经加载过也就是说 Tomcat 的类加载器是否已经加载过这个类。
如果 Tomcat 类加载器没有加载过这个类再看看系统类加载器是否加载过。
如果都没有就让ExtClassLoader去加载这一步比较关键目的防止 Web 应用自己的类覆盖 JRE 的核心类。因为 Tomcat 需要打破双亲委托机制假如 Web 应用里自定义了一个叫 Object 的类如果先加载这个 Object 类就会覆盖 JRE 里面的那个 Object 类这就是为什么 Tomcat 的类加载器会优先尝试用 ExtClassLoader 去加载因为 ExtClassLoader 会委托给 BootstrapClassLoader 去加载BootstrapClassLoader 发现自己已经加载了 Object 类直接返回给 Tomcat 的类加载器这样 Tomcat 的类加载器就不会去加载 Web 应用下的 Object 类了也就避免了覆盖 JRE 核心类的问题。
如果 ExtClassLoader 加载器加载失败也就是说 JRE 核心类中没有这类那么就在本地 Web 应用目录下查找并加载。
如果本地目录下没有这个类说明不是 Web 应用自己定义的类那么由系统类加载器去加载。这里请你注意Web 应用是通过Class.forName调用交给系统类加载器的因为Class.forName的默认加载器就是系统类加载器。
如果上述加载过程全部失败抛出 ClassNotFound 异常。
public Class<?> loadClass(String name, boolean resolve) throws ClassNotFoundException {
synchronized (getClassLoadingLock(name)) {
Class<?> clazz = null;
//1. 先在本地 cache 查找该类是否已经加载过
clazz = findLoadedClass0(name);
if (clazz != null) {
if (resolve)
resolveClass(clazz);
return clazz;
}
//2. 从系统类加载器的 cache 中查找是否加载过
clazz = findLoadedClass(name);
if (clazz != null) {
if (resolve)
resolveClass(clazz);
return clazz;
}
// 3. 尝试用 ExtClassLoader 类加载器类加载为什么
ClassLoader javaseLoader = getJavaseClassLoader();
try {
clazz = javaseLoader.loadClass(name);
if (clazz != null) {
if (resolve)
resolveClass(clazz);
return clazz;
}
} catch (ClassNotFoundException e) {
// Ignore
}
// 4. 尝试在本地目录搜索 class 并加载
try {
clazz = findClass(name);
if (clazz != null) {
if (resolve)
resolveClass(clazz);
return clazz;
}
} catch (ClassNotFoundException e) {
// Ignore
}
// 5. 尝试用系统类加载器 (也就是 AppClassLoader) 来加载
try {
clazz = Class.forName(name, false, parent);
if (clazz != null) {
if (resolve)
resolveClass(clazz);
return clazz;
}
} catch (ClassNotFoundException e) {
// Ignore
}
}
//6. 上述过程都加载失败抛出异常
throw new ClassNotFoundException(name);
}
tomcat的类加载findClass过程
public Class<?> findClass(String name) throws ClassNotFoundException {
...
Class<?> clazz = null;
try {
//1. 先在 Web 应用目录下查找类
clazz = findClassInternal(name);
} catch (RuntimeException e) {
throw e;
}
if (clazz == null) {
try {
//2. 如果在本地目录没有找到交给父加载器去查找
clazz = super.findClass(name);
} catch (RuntimeException e) {
throw e;
}
//3. 如果父类也没找到抛出 ClassNotFoundException
if (clazz == null) {
throw new ClassNotFoundException(name);
}
return clazz;
}
findClass方法大致步骤
先在 Web 应用本地目录下查找要加载的类。
如果没有找到交给父加载器去查找它的父加载器就是上面提到的系统类加载器 AppClassLoader。
如何父加载器也没找到这个类抛出 ClassNotFound 异常。
Tomcat是如何打破双亲委派机制的呢
Tomcat是先去本地目录加载为了避免本地目录覆盖掉JRE的核心类如java.lang包等先尝试用ExtClassLoader加载这样即能打破双亲委派机制有避免了覆盖掉核心类。
为什么不是尝试用AppClassLoader加载呢
如果是尝试用AppClassLoader这样不又变会双亲委派机制了嘛。