Tomcat配置文件Server.xml解析
一、Sax的事件驱动模型
-
类图
-
编辑
-
基础实现类
- DefaultHandler2: 此类扩展了SAX2基本处理程序类,以支持SAX2 LexicalHandler , DeclHandler和EntityResolver2扩展
- DefaultHandler: SAX2事件处理程序的默认基类。此类可用作SAX2应用程序的便捷基类: 它为四个核心SAX2处理程序类中的所有回调提供默认实现: EntityResolver DTDHandler ContentHandler ErrorHandler
- DefaultHandler2: 此类扩展了SAX2基本处理程序类,以支持SAX2 LexicalHandler , DeclHandler和EntityResolver2扩展
-
接口
- ContentHandler: (按照先后顺序排列)
- startDocument : 接收文档开始的通知。
- startElement : 接收元素开始的通知。 解析器将在XML文档中每个元素的开头调用此方法。 每个startElement事件都会有一个对应的endElement事件(即使元素为空)。 将在相应的endElement事件之前按顺序报告元素的所有内容。
- characters : 接收字符数据的通知。 解析器将调用此方法报告字符数据的每个块。 SAX解析器可以在单个块中返回所有连续的字符数据,也可以将其拆分为几个块。
- endElement : 接收元素结束的通知。 SAX解析器将在XML文档中每个元素的末尾调用此方法
- endDocument : 接收文档结尾的通知
- ContentHandler: (按照先后顺序排列)
二、Xml解析(Catalina.parseServerXml)
- 创建Xml解析类Digester(Catalina.createStartDigester)
- 新建对象: Digester digester = new Digester();
- 添加规则(org.apache.tomcat.util.digester.Rule):
- digester.addObjectCreate("Server", "org.apache.catalina.core.StandardServer", "className");
创建对象Server - digester.addSetProperties("Server");
给Server对象设置属性 - digester.addSetNext("Server", "setServer", "org.apache.catalina.Server");
调用setServer方法把创建的Server对象赋值给上一级对象 - digester.addRule("Server/Service/Connector", new ConnectorCreateRule())
创建Connector对象.
- digester.addObjectCreate("Server", "org.apache.catalina.core.StandardServer", "className");
- 开始解析server.xml()
- 添加root对象: Catalina: digester.push(this);
- 代码解析入口: Catalina: digester.parse(inputSource)
- Digester.startDocument : 开始解析文档
- Digester.startElement : 解析文档的xml标签
- list = updateAttributes(list);
返回一个属性列表,其中包含所有传入的属性,并且属性值中形式为“ $ {xxx}”的任何文本都将替换为系统属性中的适当值。 - List
rules = getRules().match(namespaceURI, match); matches.push(rules);
把规则列表添加到栈顶 然后再针对所有相关规则触发’begin’方法
- list = updateAttributes(list);
- Digester.characters : 解析标签之间的内容
- bodyText.append(buffer, start, length); 处理从XML元素的主体接收到的字符数据的通知
- Digester.endElement : xml标签解析结束
- bodyText = updateBodyText(bodyText);
返回一个新的StringBuilder,其中包含与输入缓冲区相同的内容,但格式$ {varname}的数据已被系统属性中定义的var值替换。 - List
rules = matches.pop();
从栈顶获取rules 然后再针对所有相关规则触发’body’方法 最后再以相反的顺序触发所有相关规则的’end’方法 - match = match.substring(0, slash);
恢复先前的匹配表达式
- bodyText = updateBodyText(bodyText);
- Digester.endDocument : 整个文档解析结束
- pop(); 从堆栈中弹出顶部对象(Catalina),然后将其返回。 如果堆栈上没有对象,则返回null
- getRules().rules().forEach(k→k.finish()); 为所有已定义的规则触发’finish’方法