一、catalina为什么不new出来?
掌握了java的类加载器和双亲委派机制,现在我们就可以回答正题上来了,tomcat的类加载器是怎么设计的?
1.web容器的特性
web容器有其自身的特殊性,所以在类加载器这块是不能完全使用jvm的类加载器的双亲委派机制的。在web容器中我们应该要满足如下的特性:
隔离性:
部署在同一个web容器上的两个web应用程序所使用的java类库可以实现相互隔离。设想一下,两个web应用,一个使用了spring3.0,另一个使用了新的的5.0,应用服务器使用一个类加载器,web应用将会因为jar包覆盖而无法启动。
灵活性:
web应用之间的类加载器相互独立,那么就能针对一个web应用进行重新部署,此时web应用的类加载器会被重建,而且不会影响其他的web应用。如果采用一个类加载器,类之间的依赖是杂乱复杂的,无法完全移出某个应用的类。
性能:
性能也是一个比较重要的点。部署在同一个web容器上的两个web应用程序所使用的java类库可以互相共享。这个需求也很常见,例如,用户可能有10个使用spring框架的应用程序部署在同一台服务器上,如果把10份spring分别存放在各个应用程序的隔离目录中,将会是很大的资源浪费——这主要倒不是浪费磁盘空间的问题,而是指类库在使用时都要被加载到web容器的内存,如果类库不能共享,虚拟机的方法区就会很容易出现过度膨胀的风险。
2.tomcat类加载器结构
明白了web容器的类加载器有多个,再来看tomcat的类加载器结构。
首先上张图,整体看下tomcat的类加载器:
可以看到在原先的java类加载器基础上,tomcat新增了几个类加载器,包括3个基础类加载器和每个web应用的类加载器,其中3个基础类加载器可在conf/catalina.properties中配置,具体介绍下:
common:
以应用类加载器为父类,是tomcat顶层的公用类加载器,其路径由conf/catalina.properties中的common.loader指定,默认指向${catalina.base}/lib下的包。
catalina:
以common类加载器为父类,是用于加载tomcat应用服务器的类加载器,其路径由server.loader指定,默认为空,此时tomcat使用common类加载器加载应用服务器。
shared:
以common类加载器为父类,是所有web应用的父类加载器,其路径由shared.loader指定,默认为空,此时tomcat使用common类加载器作为web应用的父加载器。
web应用:
以shared类加载器为父类,加载/web-inf/classes目录下的未压缩的class和资源文件以及/web-inf/lib目录下的jar包,该类加载器只对当前web应用可见,对其他web应用均不可见。
默认情况下,common、catalina、shared类加载器是同一个,但可以配置3个不同的类加载器,使他们各司其职。
首先,common类加载器复杂加载tomcat应用服务器内部和web应用均可见的类,如servlet规范相关包和一些通用工具包。
其次,catalina类加载器负责只有tomcat应用服务器内部可见的类,这些类对web应用不可见。比如,想实现自己的会话存储方案,而且该方案依赖了一些第三方包,当然是不希望这些包对web应用可见,这时可以配置server.load,创建独立的catalina类加载器。
再次,shared类复杂加载web应用共享类,这些类tomcat服务器不会依赖
3.tomcat源码分析
3.1 catalinclassloader
首先来看看tomcat的类加载器的继承结构
可以看到继承的结构和我们上面所写的类加载器的结构不同。
大家需要注意双亲委派机制并不是通过继承来实现的,而是相互之间组合而形成的。
所以appclassloader没有继承自 extclassloader,webappclassloader也没有继承自appclassloader。
至于common classloader ,shared classloader,catalina classloader则是在启动时初始化的三个不同名字的urlclassloader。
先来看看bootstrap#init()方法。init方法会调用initclassloaders,同样也会将catalina classloader设置到当前线程设置到当前线程,进入initclassloaders来看看。
private void initclassloaders() { try { // 创建 commonloader catalinaloader sharedloader commonloader = createclassloader("common", null); if (commonloader == null) { // no config file, default to this loader - we might be in a 'single' env. commonloader = this.getclass().getclassloader(); } // 默认情况下 server.loader 和 shared.loader 都为空则会返回 commonloader 类加载器 catalinaloader = createclassloader("server", commonloader); sharedloader = createclassloader("shared", commonloader); } catch (throwable t) { handlethrowable(t); log.error("class loader creation threw exception", t); system.exit(1); } }
我们可以看到在initclassloaders()方法中完成了commonclassloader, catalinaclassloader,和sharedclassloader的创建,而且进入到createclassloader方法中。
可以看到这三个基础类加载器所加载的资源刚好对应conf/catalina.properties中的common.loader,server.loader,shared.loader
3.2 层次结构
common/catalina/shared classloader的创建好了之后就会维护相互之间的组合关系
其实也就是设置了父加载器
3.3 具体的加载过程
源码比较长,直接进入到 webappclassloaderbase
中的 loadclass方法
@override public class<?> loadclass(string name, boolean resolve) throws classnotfoundexception { synchronized (getclassloadinglock(name)) { if (log.isdebugenabled()) { log.debug("loadclass(" + name + ", " + resolve + ")"); } class<?> clazz = null; // log access to stopped class loader checkstateforclassloading(name); // (0) check our previously loaded local class cache // 检查webappclassloader中是否加载过此类 clazz = findloadedclass0(name); if (clazz != null) { if (log.isdebugenabled()) { log.debug(" returning class from cache"); } if (resolve) { resolveclass(clazz); } return clazz; } // (0.1) check our previously loaded class cache // 如果第一步没有找到,则继续检查jvm虚拟机中是否加载过该类 clazz = findloadedclass(name); if (clazz != null) { if (log.isdebugenabled()) { log.debug(" returning class from cache"); } if (resolve) { resolveclass(clazz); } return clazz; } // (0.2) try loading the class with the bootstrap class loader, to prevent // the webapp from overriding java se classes. this implements // srv.10.7.2 // 如果前两步都没有找到,则使用系统类加载该类(也就是当前jvm的classpath)。 // 为了防止覆盖基础类实现,这里会判断class是不是jvmse中的基础类库中类。 string resourcename = binarynametopath(name, false); classloader javaseloader = getjavaseclassloader(); boolean tryloadingfromjavaseloader; try { // use getresource as it won't trigger an expensive // classnotfoundexception if the resource is not available from // the java se class loader. however (see // https://bz.apache.org/bugzilla/show_bug.cgi?id=58125 for // details) when running under a security manager in rare cases // this call may trigger a classcircularityerror. // see https://bz.apache.org/bugzilla/show_bug.cgi?id=61424 for // details of how this may trigger a stackoverflowerror // given these reported errors, catch throwable to ensure any // other edge cases are also caught url url; if (securitymanager != null) { privilegedaction<url> dp = new privilegedjavasegetresource(resourcename); url = accesscontroller.doprivileged(dp); } else { url = javaseloader.getresource(resourcename); } tryloadingfromjavaseloader = (url != null); } catch (throwable t) { // swallow all exceptions apart from those that must be re-thrown exceptionutils.handlethrowable(t); // the getresource() trick won't work for this class. we have to // try loading it directly and accept that we might get a // classnotfoundexception. tryloadingfromjavaseloader = true; } if (tryloadingfromjavaseloader) { try { clazz = javaseloader.loadclass(name); if (clazz != null) { if (resolve) { resolveclass(clazz); } return clazz; } } catch (classnotfoundexception e) { // ignore } } // (0.5) permission to access this class when using a securitymanager if (securitymanager != null) { int i = name.lastindexof('.'); if (i >= 0) { try { securitymanager.checkpackageaccess(name.substring(0,i)); } catch (securityexception se) { string error = sm.getstring("webappclassloader.restrictedpackage", name); log.info(error, se); throw new classnotfoundexception(error, se); } } } // 检查是否 设置了delegate属性,设置为true,那么就会完全按照jvm的"双亲委托"机制流程加载类。 boolean delegateload = delegate || filter(name, true); // (1) delegate to our parent if requested if (delegateload) { if (log.isdebugenabled()) { log.debug(" delegating to parent classloader1 " + parent); } try { clazz = class.forname(name, false, parent); if (clazz != null) { if (log.isdebugenabled()) { log.debug(" loading class from parent"); } if (resolve) { resolveclass(clazz); } return clazz; } } catch (classnotfoundexception e) { // ignore } } // (2) search local repositories // 若是没有委托,则默认会首次使用webappclassloader来加载类。通过自定义findclass定义处理类加载规则。 // findclass()会去web-inf/classes 目录下查找类。 if (log.isdebugenabled()) { log.debug(" searching local repositories"); } try { clazz = findclass(name); if (clazz != null) { if (log.isdebugenabled()) { log.debug(" loading class from local repository"); } if (resolve) { resolveclass(clazz); } return clazz; } } catch (classnotfoundexception e) { // ignore } // (3) delegate to parent unconditionally // 若是webappclassloader在/web-inf/classes、/web-inf/lib下还是查找不到class, // 那么无条件强制委托给system、common类加载器去查找该类。 if (!delegateload) { if (log.isdebugenabled()) { log.debug(" delegating to parent classloader at end: " + parent); } try { clazz = class.forname(name, false, parent); if (clazz != null) { if (log.isdebugenabled()) { log.debug(" loading class from parent"); } if (resolve) { resolveclass(clazz); } return clazz; } } catch (classnotfoundexception e) { // ignore } } } throw new classnotfoundexception(name); }
web应用类加载器默认的加载顺序是:
- (1).先从缓存中加载;
- (2).如果没有,则从jvm的bootstrap类加载器加载;
- (3).如果没有,则从当前类加载器加载(按照web-inf/classes、web-inf/lib的顺序);
- (4).如果没有,则从父类加载器加载,由于父类加载器采用默认的委派模式,所以加载顺序是appclassloader、common、shared。
tomcat提供了delegate属性用于控制是否启用java委派模式,默认false(不启用),当设置为true时,tomcat将使用java的默认委派模式,这时加载顺序如下:
- (1).先从缓存中加载;
- (2).如果没有,则从jvm的bootstrap类加载器加载;
- (3).如果没有,则从父类加载器加载,加载顺序是appclassloader、common、shared。
- (4).如果没有,则从当前类加载器加载(按照web-inf/classes、web-inf/lib的顺序)
以上就是tomcat catalina为什么不new出来原理解析的详细内容,更多关于tomcat catalina原理的资料请关注代码网其它相关文章!
发表评论