tomcat类加载器构造

tomcat类加载器结构.
                   bootstrap classloader(载入jvm自带的类和$java_home/jre/lib/ext/*.jar)
                     |
                     |
               system classloader 载入$classpath/*.class
                     |
                     |
               common classloader 载入$catalina_home/common/*,它们对tomcat和所有的web应用程序都可见
                  /          \
                 /            \
   catalina classloader(1)      shared classloader (2)
                                 /        \
                                /          \
                      webapp classloader(3)   webapp classloader

(1) 载入$catalina_home/server/*,它们仅对tomcat可见,对所有的web应用程序都不可见
(2) 载入$catalina_home/shared/*,它们仅对所有web应用程序可见,对tomcat不可见(也不必见)
(3) 载入contextbase/web-inf/*,它们仅对该web应用程序可见

classloader的原理
每个运行中的线程都有一个成员contextclassloader,用来在运行时动态地载入其它类。系统默认的contextclassloader是systemclassloader,所以一般而言java程序在执行时可以使用jvm自带的类、$java_home/jre/lib/ext/中的类和$classpath/中的类。
可以使用thread.currentthread().setcontextclassloader(...)来更改当前线程的contextclassloader,来改变其载入类的行为。
classloader被组织成树形,一般的工作原理是:
1) 线程需要用到某个类,于是contextclassloader被请求来载入该类。
2) contextclassloader请求它的父classloader来完成该载入请求。
3) 如果父classloader无法载入类,则contextclassloader试图自己来载入。
注意:webappclassloader的工作原理和上述有少许不同:
它先试图自己载入类(在contextbase/web-inf/中载入类),如果无法载入,再请求父classloader完成,由此可得:
- 对于web应用程序线程,它的contextclassloader是webappclassloader
- 对于tomcat server线程,它的contextclassloader是catalinaclassloader。