开头说两句
Java 基础 Demo 站: https://www.javastudy.cloud
Java 中高级开发博客: https://www.lixiang.red
Java 学习公众号: java 技术大本营
学习背景
在配置完 server.xml 之后,tomcat 会解析 server.xml 然后初始化 servlet 容器,在前面我们学习 tomcat 的启动过程:
https://www.lixiang.red/articles/2019/08/07/1565188609044.html
Catalina 的 start 方法
跟着昨天的脚步,在 Bootstrap 的 init 之后,后调用 Bootstrap 的 start 方法
public void start()
throws Exception {
if( catalinaDaemon==null ){
init();
}
// 使用反射去调用Catalina 的 start 方法
Method method = catalinaDaemon.getClass().getMethod("start", (Class [] )null);
method.invoke(catalinaDaemon, (Object [])null);
}
如上所示: 再通地反射去调用 Catalina.java 的 start 方法
在方法开始的地方,调用了 load 方法去加载:
if (getServer() == null) {
load();
}
load 初始化 Digester 相关信息
先是在 initNaming 设置一些运行时的属性,插入一下 ENV 和 Property 的知识点
ENV 和 Property
public class EnvStudy {
public static void main(String[] args) {
Map<String, String> getenv = System.getenv();
System.out.println(getenv);
Properties properties = System.getProperties();
System.out.println(properties);
}
}
ENV 是电脑的环境变量
Properties 是程序运行时一些属性
初始化 digester
// Before digester - it may be needed
initNaming();
// Create and execute our Digester
// 在createStartDigester里面配置server.xml的格式以及解析规则
Digester digester = createStartDigester();
createStartDigester 的代码部分如下:
通过对比前面 load 方法,我们可以看到两个计时方法
// 这个方法,是专门计时用的,其时间现现实生活中的时间,没有半毛钱关系
long t1 = System.nanoTime();
// 这个方法是和操作系统相关,对应着我们现实生活的时间
long t1=System.currentTimeMillis();
通过寻找熟悉的元素,我们发现,digester 和 server.xml 里面是一一对应的
digester.addObjectCreate("Server",
"org.apache.catalina.core.StandardServer",
"className");
如上方法所代表的意思是:
当匹配到 Server 节点时,取 className 来创建 Object , 如果设置的 className 为空,就使用 StandardServer 来创建 Object
digester 的栈结构
----- 栈顶
GlobalNamingResources
-----
Server
-----
Catalina
-----
digester.addSetNext("Server/GlobalNamingResources",
"setGlobalNamingResources",
"org.apache.catalina.deploy.NamingResourcesImpl");
找到次栈顶的元素 Server
, 调用其 setGlobalNamingResources
方法,然后把栈顶元素 GlobalNamingResources
做为参数传入到上面的方法中
启动 Servlet 容器
在经历一系列的 inputstream 判空之后, 开始进入真正启动的过程
// 使用上述定义好的digester规则去解析server.xml
inputSource.setByteStream(inputStream);
digester.push(this);
digester.parse(inputSource);
在这里要注意的是 digester.push(this), 把当前的 Catalina 实例传入到 Digester 里面做根,然后在解析 xml 时,会调用 this.setServer , 生成 StandardServer 实例
然后调用 server.init 方法,去初始化 Server. 然后在初始化 server 的时候,再初始化 GlobalNamingResources, Services 等等
今天的 server.xml 解析并初始化 servlet 容器 过种就结束啦
最后说两句
前面这些过程以解析和生成对应对象设置环境为主, 后面就真正进入的 tomcat 的 Container 的初始化, 大家在学习过程中,有什么问题,可以随时和小刀沟通交流: best396975802
欢迎来到这里!
我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。
注册 关于