[] JBoss、Tomcat Classloader不完全分析

本文详细解析了JBoss 4.2.x版本中独特的ClassLoading机制,包括UnifiedClassLoader3如何实现类共享及可能产生的版本冲突问题。同时对比介绍了Tomcat 6/7的ClassLoader层次结构与加载流程。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

[b][size=medium]由于平时项目中用到的还是jboss 4.2.x所以我这里的分析时针对这个版本的,不一定适用其他jboss版本。[/size][/b]
下面言归正传。
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。
这几个对象的关系请见下图
[img]http://seanhe.iteye.com/upload/picture/pic/77935/6bc53ba0-942a-3a51-ab34-8910c2465e4c.png[/img]
从上图可以看出默认情况下所有的ucl共享一个repository,通过repository可以实现class的共享。unifiedloaderrepository3实例中维护着两个容器,一个是class cache:这个容器很明显缓存了一些class,这样可以提高loadclass方法的执行效率;另一个是packagesmap:这个map维护的是类的包名和ucl的mapping关系。具体ucl按什么逻辑load class的请看下面的活动图:
[img]http://seanhe.iteye.com/upload/picture/pic/77937/9dc09782-b84c-31a2-96ef-f6e09050f9e9.jpg[/img]
unifiedclassloader3的继承结构如下图所示,unifiedclassloader3的父类repositoryclassloader重写了urlclassloader的loadclass方法,实现了上图的逻辑
[img]http://seanhe.iteye.com/upload/picture/pic/77965/aad0ece3-2a36-320c-a8d8-16d6b11e7485.jpg[/img]
下面请看一下相关代码:
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");         }      }   }

有几点结论可以加深一些印象:
[list=1][*]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来加载
[/list]
[color=red]由于jboss对所有ucl的共享机制就会导致出现一些class的版本冲突问题,一些应用加载不到自己的应用需要的class。对于这个问题jboss提供了一些解决措施:[url=http://community.jboss.org/wiki/classloadingconfiguration]http://community.jboss.org/wiki/classloadingconfiguration[/url]。后面我会再整理一下此前我遇到过的一个class冲突的case和解决办法。[/color]

[b][size=medium]tomcat6/7 class loader机制[/size][/b]
tomcat class loader层次结构
[img]http://seanhe.iteye.com/upload/picture/pic/77967/f46421a3-35c3-3047-b353-49ef978af8c9.jpg[/img]
tomcat的class load机制较jboss来说要简单
当web应用需要load class时先调用webappclassloader的loadclass方法,loadclass内部逻辑如下:
[list=1][*]调用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步。
[/list]
具体的代码可以看webappclassloader.loadclass(..)

参考文档:
[url=http://community.jboss.org/wiki/jbossclassloadingusecases]http://community.jboss.org/wiki/jbossclassloadingusecases[/url]
[url=http://community.jboss.org/wiki/classloadingconfiguration]http://community.jboss.org/wiki/classloadingconfiguration[/url]
[url=http://community.jboss.org/wiki/enableclassloaderlogging]http://community.jboss.org/wiki/enableclassloaderlogging[/url]
[url=http://agapple.iteye.com/blog/791940]http://agapple.iteye.com/blog/791940[/url]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值