【tcomat】聊聊tomcat是如何打破双亲委派模型进行类加载的
    		       		warning:
    		            这篇文章距离上次修改已过425天,其中的内容可能已经有所变动。
    		        
        		                
                Tomcat打破双亲委派模型的核心在于它自定义的类加载器结构。Tomcat 有多个类加载器,它们具有特定的加载范围,用以隔离不同应用之间的类库,保证应用的独立性。
Tomcat 类加载器层次结构如下:
CommonClassLoader:加载Tomcat的核心类库,如servlet-api等。CatalinaClassLoader:为Tomcat的核心类和连接器(connectors)提供类加载空间,不加载应用程序的类。SharedClassLoader:加载共享库,对所有应用程序可见,但不是应用程序专用。WebappClassLoader:加载特定Web应用程序的类,隔离不同应用的类加载。
WebappClassLoader 继承了 StandardClassLoader,它在双亲委派模型基础上,通过覆盖 loadClass 方法实现了类的重写加载。
以下是一个简化的 WebappClassLoader 的 loadClass 方法示例:
public class WebappClassLoader extends URLClassLoader {
    // ...
 
    @Override
    protected Class<?> loadClass(String name, boolean resolve) throws ClassNotFoundException {
        synchronized (getClassLoadingLock(name)) {
            // 首先尝试从本地缓存加载类
            Class<?> cls = findLoadedClass(name);
            if (cls == null) {
                // 如果本地缓存中没有,尝试从父加载器(WebappClassLoader的父加载器可能是SharedClassLoader或CatalinaClassLoader)加载
                try {
                    if (securityManager != null) {
                        checkPackageAccess(name);
                    }
                    cls = findClass(name);
                } catch (ClassNotFoundException e) {
                    // 父加载器无法加载时,尝试从本地资源中加载
                    cls = findClass(name);
                }
            }
            if (resolve) {
                resolveClass(cls);
            }
            return cls;
        }
    }
 
    // ...
}在这个示例中,loadClass 方法首先检查是否已经加载过该类,如果没有加载过,它将尝试使用父加载器来加载类。如果父加载器抛出 ClassNotFoundException 异常,表明它无法加载这个类,WebappClassLoader 实例将尝试从其类路径中查找并加载这个类。这样做打破了双亲委派模型,允许 WebappClassLoader 加载应用专有的类而不影响Tomcat的核心类库。
评论已关闭