Tomcat中一个web应用程序对应一个Context容器,Context容器启动的主要逻辑在StandardContext类的start方法中。
start方法中首先会执行init方法,Context容器对应的监听器是ContextConfig,产生的init事件交由ContextConfig的init方法处理:
1、读取conf/context.xml配置文件,并解析它。
2、读取/conf/Catalina/localhost/context.xml.default配置文件,如果存在的话,则解析它。
3、如果web应用是war包的形式,则进行解压操作,Host容器进行部署web应用解压war包的操作就是这里进行的。
4、获取web应用实际的物理路径,并将路径赋值给Context容器,设置为应用的工作目录。
接着会产生start事件,同样,start事件也会由ContextConfig的start方法处理:
1、读取conf/web.xml配置文件,如果存在的话则解析它。这个文件中配置了两个Servlet,分别是JspServlet和DefaultServlet,并且它们的load-on-startup属性值都大于-1。关于load-on-startup属性的作用后面会有讲解。
2、读取/WEB-INF/web.xml配置文件,如果存在的话则解析它。这个配置文件就是我们在进行web开发时常见的web.xml文件了。
start事件处理完后会创建StandardManager对象。StandardManager对象是Manage接口的标准实现类,负责管理Session对象,比如创建和销毁Session对象。setManager方法将StandardManager对象和Context容器关联起来。关于Session更详细的部分在tomcat处理请求的部分会慢慢道来。
然后会调用filterStart方法创建ApplicationFilterConfig对象并初始化web应用的web.xml中配置的过滤器对象。关于过滤器的部分,后面会有详细的补充。
最后调用loadOnStartup实例化load-on-startup属性大于-1的Servlet对象。那么看下loadOnStartup方法的源码实现。
public void loadOnStartup(Container children[]) {
// Collect "load on startup" servlets that need to be initialized
TreeMap map = new TreeMap();
for (int i = 0; i < children.length; i++) {
Wrapper wrapper = (Wrapper) children[i];
int loadOnStartup = wrapper.getLoadOnStartup();
if (loadOnStartup < 0)
continue;
Integer key = Integer.valueOf(loadOnStartup);
ArrayList list = (ArrayList) map.get(key);
if (list == null) {
list = new ArrayList();
map.put(key, list);
}
list.add(wrapper);
}
// Load the collected "load on startup" servlets
Iterator keys = map.keySet().iterator();
while (keys.hasNext()) {
Integer key = (Integer) keys.next();
ArrayList list = (ArrayList) map.get(key);
Iterator wrappers = list.iterator();
while (wrappers.hasNext()) {
Wrapper wrapper = (Wrapper) wrappers.next();
try {
wrapper.load();
} catch (ServletException e) {
getLogger().error(sm.getString("standardWrapper.loadException",
getName()), StandardWrapper.getRootCause(e));
// NOTE: load errors (including a servlet that throws
// UnavailableException from tht init() method) are NOT
// fatal to application startup
}
}
}
}
在for循环中,根据Servlet的配置的load-on-startup属性值,如果小于大于等于0,则会被添加到list集合中。而添加到集合中的Servlet对象会调用load方法进行实例化,因此我们知道了如果Servlet对象的load-on-startup配置项大于等于0,那么在Context容器启动时就会实例化该对象。