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

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

[复制链接]

尚未签到

发表于 2017-2-1 09:49:26 | 显示全部楼层 |阅读模式
由于平时项目中用到的还是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
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层次结构
DSC0003.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

运维网声明 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-336007-1-1.html 上篇帖子: Tomcat 出现OutOfMemoryError: PermGen space 错误 下篇帖子: 应用jgroups实现tomcat session的同步
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

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

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

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

扫描微信二维码查看详情

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


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


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


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



合作伙伴: 青云cloud

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