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

[经验分享] JBoss、Tomcat Classloader不完全分析

[复制链接]

尚未签到

发表于 2015-8-9 08:06:43 | 显示全部楼层 |阅读模式
  http://seanhe.iteye.com/blog/841723
  
  由于平时项目中用到的还是JBoss 4.2.x所以我这里的分析时针对这个版本的,不一定适用其他JBoss版本。
下面言归正传。
JBoss为了实现类的共享引入了class loader repository的概念,并且设计了org.jboss.mx.loading.UnifiedClassLoader3 (UCL)来完成sharing classes的主要功能。
UCL和UnifiedLoaderRepository3 一对多的关系,默认情况下一个jboss实例中只有一个UnifiedLoaderRepository3实例,这个UnifiedLoaderRepository实例会和所有的UCL关联。
NoAnnotationClassLoader是UCL的父classloader用来加载$JBOSS_HOME/lib下的jar,system class loader是NoAnnotationClassLoader的父classloader。
这几个对象的关系请见下图
DSC0000.png
从上图可以看出默认情况下所有的UCL共享一个Repository,通过Repository可以实现class的共享。 UnifiedLoaderRepository3实例中维护着两个容器,一个是class cache:这个容器很明显缓存了一些class,这样可以提高loadClass方法的执行效率;另一个是packagesMap:这个map维护的是 类的包名和UCL的mapping关系。具体UCL按什么逻辑load class的请看下面的活动图:
DSC0001.jpg
UnifiedClassLoader3的继承结构如下图所示,UnifiedClassLoader3的父类RepositoryClassLoader重写了URLClassLoader的loadClass方法,实现了上图的逻辑
DSC0002.jpg
下面请看一下相关代码:
RepositoryClassLoader.java



Java代码   DSC0003.png

  • public Class loadClass(String name, boolean resolve)  
  •       throws ClassNotFoundException  
  •    {  
  •       boolean trace = log.isTraceEnabled();  
  •       if (trace)  
  •          log.trace("loadClass " + this + " name=" + name+", loadClassDepth="+loadClassDepth);  
  •       Class clazz = null;  
  •       try  
  •       {  
  •          if (repository != null)  
  •          {  
  •             clazz = repository.getCachedClass(name);//先从cache中load class  
  •             if (clazz != null)  
  •             {  
  •                if( log.isTraceEnabled() )  
  •                {  
  •                   StringBuffer buffer = new StringBuffer("Loaded class from cache, ");  
  •                   ClassToStringAction.toString(clazz, buffer);  
  •                   log.trace(buffer.toString());  
  •                }  
  •                return clazz;  
  •             }  
  •          }  
  •          //loadClassImpl中会调用的LoadMgr3的一些方法实现上面流程图的逻辑,具体的代码实现比较冗长,这里就不贴出来了  
  •          clazz = loadClassImpl(name, resolve, Integer.MAX_VALUE);  
  •          return clazz;  
  •       }  
  •       finally  
  •       {  
  •          if (trace)  
  •          {  
  •             if (clazz != null)  
  •                log.trace("loadClass " + this + " name=" + name + " class=" + clazz + " cl=" + clazz.getClassLoader());  
  •             else  
  •                log.trace("loadClass " + this + " name=" + name + " not found");  
  •          }  
  •       }  
  •    }  
  
有几点结论可以加深一些印象:

  • JBoss做为Application Server可以部署ear包也可以war包。但是jboss在默认情况下处理ear和war是两种class load机制。
  • 当部署ear时,JBOSS默认使用我前面提到的class load机制。一个ear包里的所有的jar由一个UCL统一加载和管理
  • 需要注意的是ear里的war的部署有点特别。它只是将自身添加到ucl的classpath域中,而war下的WEB-INF/lib/*.jar,则是由WebAppClassloader来加载
  
由于jboss对所有UCL的共享机制就会导致出现一些class的版本冲突问题,一些应用加载不到自己的应用需要的class。对于这个问题JBOSS提供了一些解决措施:http://community.jboss.org/wiki/ClassLoadingConfiguration。后面我会再整理一下此前我遇到过的一个class冲突的case和解决办法。

Tomcat6/7 class loader机制
Tomcat class loader层次结构
DSC0004.jpg
Tomcat的class load机制较Jboss来说要简单
当web应用需要load class时先调用WebAppClassloader的loadClass方法,loadClass内部逻辑如下:

  • 调用findLoadedClass(String)检查此class是否已经加载,如果以加载则直接返回,如果没有则继续做下一步。
  • 调用system class loader的loadClass的方法,这样可以加载到JDK核心的类,如果没有找到符合的类则继续做下一步。
  • 如果WebAppClassloader的delegate属性是true或者正在处理的class在过滤列表里,会先从父class loader中加载类。如果没有则继续做下一步。一般这一步不会执行。
  • WebAppClassloader在自己的类库(WEB-INF/classes和WEB-INF/lib)中查找class。如果没有则继续做下一步。
  • 如果前面第3步已经跳过这一步会继续执行。如果前面第3步已经执行过,这一步就不会再执行。这一步的执行逻辑同第3步。
  
具体的代码可以看WebAppClassloader.loadClass(..)
  
  
参考文档:
http://community.jboss.org/wiki/JBossClassLoadingUseCases
http://community.jboss.org/wiki/ClassLoadingConfiguration
http://community.jboss.org/wiki/EnableClassloaderLogging
http://agapple.iteye.com/blog/791940
  http://www.iteye.com/topic/826661

运维网声明 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-96202-1-1.html 上篇帖子: Eclipse运行Tomcat第一个小程序 下篇帖子: j2ee、mvn、eclipse、Tomcat等中文乱码问题解决方法
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

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

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

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

扫描微信二维码查看详情

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


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


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


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



合作伙伴: 青云cloud

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