设为首页 收藏本站
查看: 290|回复: 0

[经验分享] Tomcat的WebappClassLoader(web应用类加载器)详解 (转)

[复制链接]

尚未签到

发表于 2017-2-7 08:06:35 | 显示全部楼层 |阅读模式
转自:http://blog.sina.com.cn/s/blog_4d398d670100dmui.html
Tomcat负责Web应用的类加载的是org.apache.catalina.loader.WebappClassLoader,它有几个比较重要的方法:findClass(),loadClass(),findClassInternal(),findResourceInternal().
 
 WebappClassLoader类加载器被用来加载一个类的时候,loadClass()会被调用,loadClass()则调用findClass()。后两个方法是WebappClassLoader的私有方法,findClass()调用findClassInternal()来创建class对象,而findClassInternal()则需要findResourceInternal()来查找.class文件。
 
通常自己实现类记载器只要实现findclass即可,这里为了实现特殊目的而override了loadClass().
 
下面是精简过的代码(去除了几乎全部关于log、异常和安全控制的代码):
 
loadClass:
    public Class loadClass(String name, boolean resolve)
            throws ClassNotFoundException {
        Class clazz = null;
        // (0) 先从自己的缓存中查找,有则返回,无则继续
        clazz = findLoadedClass0(name);
        if (clazz != null) {
            if (resolve) resolveClass(clazz);            
            return (clazz);
        }
        // (0.1) 再从parent的缓存中查找
        clazz = findLoadedClass(name);
        if (clazz != null) {
            if (resolve) resolveClass(clazz);
            return (clazz);
        }
        // (0.2) 缓存中没有,则首先使用system类加载器来加载
        clazz = system.loadClass(name);
         if (clazz != null) {
             if (resolve) resolveClass(clazz);
             return (clazz);
         }
        //判断是否需要先让parent代理
        boolean delegateLoad = delegate || filter(name);
        // (1) 先让parent加载,通常delegateLoad == false,即这一步不会执行
        if (delegateLoad) {
            ClassLoader loader = parent;
            if (loader == null)
                loader = system;
            clazz = loader.loadClass(name);
            if (clazz != null) {
                if (resolve) resolveClass(clazz);
                return (clazz);
            }
        }
        // (2) delegateLoad == false 或者 parent加载失败,调用自身的加载机制
        clazz = findClass(name);
        if (clazz != null) {
            if (resolve) resolveClass(clazz);
            return (clazz);
        }
        // (3) 自己加载失败,则请求parent代理加载
        if (!delegateLoad) {
            ClassLoader loader = parent;
            if (loader == null)
                loader = system;
            clazz = loader.loadClass(name);
            if (clazz != null) {
                return (clazz);
            }
        }
        throw new ClassNotFoundException(name);
    }
 
 
findClass:
 
public Class findClass(String name) throws ClassNotFoundException {
        // 先试图自己加载类,找不到则请求parent来加载
        // 注意这点和java默认的双亲委托模式不同
        Class clazz = null;
        clazz = findClassInternal(name);
        if ((clazz == null) && hasExternalRepositories) {
            synchronized (this) {
                clazz = super.findClass(name);
            }
        }
        if (clazz == null) {
            throw new ClassNotFoundException(name);
        }
        return (clazz);
    }
 

 findClassInternal:
 
protected Class findClassInternal(String name)
        throws ClassNotFoundException {
        if (!validate(name))
            throw new ClassNotFoundException(name);
        //根据类名查找资源
        String tempPath = name.replace('.', '/');
        String classPath = tempPath + ".class";
        ResourceEntry entry = null;
        entry = findResourceInternal(name, classPath);

        if (entry == null)
            throw new ClassNotFoundException(name);
        //如果以前已经加载成功过这个类,直接返回
        Class clazz = entry.loadedClass;
        if (clazz != null)
            return clazz;
        //以下根据找到的资源(.class文件)进行:1、定义package;2、对package安全检查;3、定义class,即创建class对象
        synchronized (this) {
            if (entry.binaryContent == null && entry.loadedClass == null)
                throw new ClassNotFoundException(name);
            // Looking up the package
            String packageName = null;
            int pos = name.lastIndexOf('.');
            if (pos != -1)
                packageName = name.substring(0, pos);
            Package pkg = null;
            if (packageName != null) {
                pkg = getPackage(packageName);
                // Define the package (if null)
                if (pkg == null) {
                    //定义package的操作,此处省略,具体参看源码 
                    pkg = getPackage(packageName);
                }
            }
            if (securityManager != null) {
                //安全检查操作,此处省略,具体参看源码 
            }
            //创建class对象并返回
            if (entry.loadedClass == null) {
                try {
                    clazz = defineClass(name, entry.binaryContent, 0,
                            entry.binaryContent.length,
                            new CodeSource(entry.codeBase, entry.certificates));
                } catch (UnsupportedClassVersionError ucve) {
                    throw new UnsupportedClassVersionError(
                            ucve.getLocalizedMessage() + " " +
                            sm.getString("webappClassLoader.wrongVersion",
                                    name));
                }
                entry.loadedClass = clazz;
                entry.binaryContent = null;
                entry.source = null;
                entry.codeBase = null;
                entry.manifest = null;
                entry.certificates = null;
            } else {
                clazz = entry.loadedClass;
            }
        }
        return clazz;
    }
 
 findResouceInternal():
//要先加载相关实体资源(.jar) 再加载查找的资源本身
 
protected ResourceEntry findResourceInternal(String name, String path) {
 
//先根据类名从缓存中查找对应资源 ,有则直接返回
ResourceEntry entry = (ResourceEntry) resourceEntries.get(name);
if (entry != null) return entry; int contentLength = -1;
 
//资源二进制数据长度
InputStream binaryStream = null;
//资源二进制输入流
int jarFilesLength = jarFiles.length;
//classpath中的jar包个数
int repositoriesLength = repositories.length;
//仓库数(classpath每一段称为repository仓库)
int i; Resource resource = null;
//加载的资源实体
boolean fileNeedConvert = false;
//对每个仓库迭代,直接找到相应的entry,如果查找的资源是一个独立的文件,在这个代码块可以查找到相应资源,
//如果是包含在jar包中的类,这段代码并不能找出其对应的资源
for (i = 0; (entry == null) && (i < repositoriesLength); i++) {
  try {
         String fullPath = repositories + path;
        //仓库路径 加资源路径得到全路径
        Object lookupResult = resources.lookup(fullPath);
       //从资源库中查找资源
        if (lookupResult instanceof Resource) {
               resource = (Resource) lookupResult;
         }
        //到这里没有抛出异常,说明资源已经找到,现在构造entry对象
        if (securityManager != null) {
              PrivilegedAction dp = new PrivilegedFindResource(files, path);
             entry = (ResourceEntry)AccessController.doPrivileged(dp);
        } else {
             entry = findResourceInternal(files, path);
            //这个方式只是构造entry并给其codebase和source赋值
        }
          //获取资源长度和最后修改时间
        ResourceAttributes attributes = (ResourceAttributes) resources.getAttributes(fullPath);
        contentLength = (int) attributes.getContentLength();
        entry.lastModified = attributes.getLastModified();
         //资源找到,将二进制输入流赋给binaryStream
         if (resource != null) {
               try {
                      binaryStream = resource.streamContent();
               } catch (IOException e) {
                       return null;
               }
          //将资源的最后修改时间加到列表中去,代码略去,参加源码
         }
     } catch (NamingException e) { }
}
if ((entry == null) && (notFoundResources.containsKey(name)))
        return null;
 
//开始从jar包中查找
  JarEntry jarEntry = null;
 
synchronized (jarFiles) {
     if (!openJARs()) { return null; }
     for (i = 0; (entry == null) && (i < jarFilesLength); i++) {
          jarEntry = jarFiles.getJarEntry(path);
          //根据路径从jar包中查找资源
          //如果jar包中找到资源,则定义entry并将二进制流等数据赋给entry,同时将jar包解压到workdir.
     if (jarEntry != null) {
           entry = new ResourceEntry();
            try {
                entry.codeBase = getURL(jarRealFiles, false);
                String jarFakeUrl = getURI(jarRealFiles).toString();
                jarFakeUrl = "jar:" + jarFakeUrl + "!/" + path;
                entry.source = new URL(jarFakeUrl);
                entry.lastModified = jarRealFiles.lastModified();
             } catch (MalformedURLException e) {
                  return null;
              }
            contentLength = (int) jarEntry.getSize();
            try {
                  entry.manifest = jarFiles.getManifest();
                  binaryStream = jarFiles.getInputStream(jarEntry);
             } catch (IOException e) { return null; }
 
          if (antiJARLocking && !(path.endsWith(".class"))) {
           //解压jar包代码,参见源码 }
        }
      }
     if (entry == null) { synchronized (notFoundResources) { notFoundResources.put(name, name); } return null; } //从二进制流将资源内容读出 if (binaryStream != null) { byte[] binaryContent = new byte[contentLength]; //读二进制流的代码,这里省去,参见源码 entry.binaryContent = binaryContent; } } // 将资源加到缓存中 synchronized (resourceEntries) { ResourceEntry entry2 = (ResourceEntry) resourceEntries.get(name); if (entry2 == null) { resourceEntries.put(name, entry); } else { entry = entry2; } } return entry; }

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-338519-1-1.html 上篇帖子: Tomcat的WebappClassLoader(web应用类加载器)详解 下篇帖子: java web项目部署到tomcat时报错
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表