Tomcat源码的catalina中利用Digester解析conf/server.xml

最近在学习Tomcat的源码,在catalina. createStartDigester方法中,Tomcat开发人员采用了Digester来读取conf/server.xml文件,以前读取xml文件一般采用Dom4jSAX。由于对Digester比较陌生,所以今天抽时间研究了一下Digester是如何解析xml文件的。先简单阐述下Dom4jSAX解析XML的却别:

Dom4j是把一个xml文件全部读取到内存中,构建成一个DOM树来解析,所以Dom4j适合读取比较小的xml文件。

SAX是基于文件流来解析xml文件的,在读取xml文件流时,SAX会通过节点来触发相应的操作,也可以说SAX是基于文件流的事情触发机制来解析xml文件的。

Digeterapachecommon项目,作用是将XML转化成对象,使用者直接从对象中获取xml的节点信息。Digester是对SAX的包装,它也是基于文件流来解析xml文件,只不过这些解析操作对用户是透明的。Tomcat的配置文件conf/server.xml就是用Digester来读取的。

一、    下载Digesterjar

(1)      Digesterjar下载地址(版本:2.0
http://commons.apache.org/digester/

(2)      Digester依赖的Loggingjar下载地址(版本:1.1.1

http://commons.apache.org/logging/

(3)      Digester依赖的BeanUtilsjar下载地址(版本:1.8.3

http://commons.apache.org/beanutils/

 

只需要按照以上步骤下载jar包后,导入到eclipse工程即可。为了大家省去jar包下载的麻烦,我在博客资源中上传了“Digester_jar”资源,这里面包含了所有需要的jar包和源码。

对应的对象的源码参考我博客资源中的“Digester_object”资源,这里包含了测试代码中需要的Java对象代码。

二、    关键方法说明

(1)    serverDigester.addObjectCreate("Server","com.test.server.digester.Server")

当解析xml文件时,遇到“Server”就初始化一个“com.test.server.digester.Server”对象,并且把该对象压入栈顶

(2)    serverDigester.addSetProperties("Server", "port", "port")

Server对象注册port属性,当解析到Server节点的port属性时调用ServersetPort方法

(3)    serverDigester.addSetNext("Server/Listener", "addListener","com.test.server.digester.Listener")

当解析Server节点下的Listener节点的时候,调用Server对象的addListener方法,把当前Listener对象写入到Server对象中。无论Server节点下有多少个Listener节点,都会调用addListener方法

(4)    serverDigester.addCallMethod("Server/Service/Engine", "setEngine", 0)

Service中添加Engine,调用当前top objectsetEngine函数,参数个数为0

addCallMethodaddBeanPropertySetter方法等价

三、    注意事项

Xml中定义的属性要与对象中setget方法一致,比如xml中定义了一个节点属性“engineType”,那么在java对象的set方法必须为setEngineType,除了首字母外,其他字母必须一致,不然在解析的时候会解析不到对应的属性值。不知道是否还有其他规则,没有摸清楚。

四、    XML解析代码

<?xml version="1.0" encoding="UTF-8" ?>

<!--

         Attribute must be lower case.

         For example:

                  attribute :engineType

                  Method in Service must be setEngineType(All attribute character but one must be the same with method in PO!)

 

-->

<Server port="9996" debug="0">

  <Listener className="org.apache.catalina.mbeans.ServerLifecycleListener"

            debug="0"/>

  <Listener className="org.apache.catalina.mbeans.GlobalResourcesLifecycleListener"

            debug="1"/>

  <Service name="Catalina">

    <Engine engineType="12">I am a Engine</Engine>

  </Service>

</Server>

 

五、    Java对象代码(也可以见博客资源“Digester_object

(1)    Server.java代码

/**

 *

 */

package com.test.server.digester;

 

import java.util.Vector;

 

/**

 * 模仿解析Tomcatconf/server.xml中的节点serve

 *

 * @author rey

 *

 */

public class Server {

 

         /**

          *构造函数

          */

         public Server() {

                  super();

         }

 

         // 成员变量===============================

         private int m_nPort = 0;

         private int m_nIsDebug = 0;

         private Service m_Service = null;

         private Vector m_listenerVector = new Vector();

 

         // 公有方法=================================

         /**

          * @return the m_nPort

          */

         public int getPort() {

                  return m_nPort;

         }

 

         /**

          * @param mNPort

          *            the m_nPort to set

          */

         public void setPort(int mNPort) {

                  m_nPort = mNPort;

         }

 

         /**

          * @return the m_nIsDebug

          */

         public int getIsDebug() {

                  return m_nIsDebug;

         }

 

         /**

          * @param mNIsDebug

          *            the m_nIsDebug to set

          */

         public void setIsDebug(int mNIsDebug) {

                  m_nIsDebug = mNIsDebug;

         }

 

         /**

          * @return the m_service

          */

         public Service getService() {

                  return m_Service;

         }

 

         /**

          * @param mService the m_service to set

          */

         public void setService(Service mService) {

                  m_Service = mService;

         }

 

         public void addListener(Listener _listener) {

                  m_listenerVector.add(_listener);

         }

 

         public Vector getListeners() {

                  return m_listenerVector;

         }

}

 

(2)    Service.java代码

/**

 *

 */

package com.test.server.digester;

 

/**

 * 模仿解析Tomcatconf/server.xml中的节点Service

 *

 * @author rey

 *

 */

public class Service {

 

         /**

          *

          */

         public Service() {

                  super();

         }

 

         // 成员变量===============================

         private String m_sName = null;

         private String m_sEngine =null;

         private String m_sEngineType=null;

 

 

         // 公有方法=================================

 

         /**

          * @return the m_sEngineType

          */

         public String getEngineType() {

                  return m_sEngineType;

         }

 

         /**只能有第一个字母大写,如果还有其他的大写,则解析不出来

          * @param mSEngineType the m_sEngineType to set

          */

         public void setEngineType(String mSEngineType) {

                  m_sEngineType = mSEngineType;

         }

 

         /**

          * @return the m_sEngine

          */

         public String getEngine() {

                  return m_sEngine;

         }

 

         /**

          * @param mSEngine the m_sEngine to set

          */

         public void setEngine(String mSEngine) {

                  m_sEngine = mSEngine;

         }

        

         /**

          * @return the m_sName

          */

         public String getName() {

                  return m_sName;

         }

 

 

         /**

          * @param mSName

          *            the m_sName to set

          */

         public void setName(String mSName) {

                  m_sName = mSName;

         }

}

(3)    Listener代码

/**

 *

 */

package com.test.server.digester;

 

/**

 * 模仿解析Tomcatconf/server.xml中的节点Listener

 *

 * @author rey

 *

 */

public class Listener {

 

         /**

          *

          */

         public Listener() {

                  super();

         }

 

         // 成员变量===============================

 

         private String m_sClassName = null;

         private int m_nIsDebug = 0;

 

         // 公有方法=================================

         /**

          * @return the m_nIsDebug

          */

         public int getDebug() {

                  return m_nIsDebug;

         }

 

         /**

          * @param mNIsDebug

          *            the m_nIsDebug to set

          */

         public void setDebug(int mNIsDebug) {

                  m_nIsDebug = mNIsDebug;

         }

 

         /**

          * @return the m_sClassName

          */

         public String getClassName() {

                  return m_sClassName;

         }

 

         /**

          * @param mSClassName

          *            the m_sClassName to set

          */

         public void setClassName(String mSClassName) {

                  m_sClassName = mSClassName;

         }

}

 

(4)    Engine代码

/**

 *

 */

package com.test.server.digester;

 

/**

 * 模仿解析Tomcatconf/server.xml中的节点Engine

 * @author rey

 *

 */

public class Engine {

 

         /**

          *

          */

         public Engine() {

                  super();

         }

        

         // 成员变量===============================

         private String m_sName = null;

         private int m_nIsDebug = 0;

        

         // 公有方法=================================

         /**

          * @return the m_sName

          */

         public String getName() {

                  return m_sName;

         }

 

         /**

          * @param mSName

          *            the m_sName to set

          */

         public void setName(String mSName) {

                  m_sName = mSName;

         }

         /**

          * @return the m_nIsDebug

          */

         public int getIsDebug() {

                  return m_nIsDebug;

         }

 

         /**

          * @param mNIsDebug

          *            the m_nIsDebug to set

          */

         public void setIsDebug(int mNIsDebug) {

                  m_nIsDebug = mNIsDebug;

         }

}

 

六、    Java解析xml代码

DigesterServer代码:

/**

 *

 */

package com.test.server.digester;

 

import java.io.FileInputStream;

import java.io.IOException;

import java.util.Vector;

 

import org.apache.commons.digester.Digester;

import org.xml.sax.SAXException;

 

/**

 * 用于解析Tomcatconf/server.xmlDiagester

 *

 * @author rey

 *

 */

public class DigesterServer {

 

         /**

          * @param args

          * @throws SAXException

          * @throws IOException

          */

         public static void main(String[] args) throws IOException, SAXException {

                  String sXMLFile = "D://temp//test_diagester.xml";

                  Digester serverDigester = new Digester();

                  FileInputStream inputXMLStream = new FileInputStream(sXMLFile);

 

                  // 解析Server.xml

                  serverDigester.setValidating(false);

 

                  // 注册Server规则,解析XML时,遇到Server,就实例化一个com.test.server.digester.Server对象,并且压栈

                  serverDigester.addObjectCreate("Server",

                                   "com.test.server.digester.Server");

                 

                  //Server对象设置一个属性,可以调用setPort进行设置

                  serverDigester.addSetProperties("Server", "port", "port");

                  serverDigester.addSetProperties("Server", "debug", "debug");

 

                  // 注册Listener规则,这里的斜线是根据xmlpath来计算的

                  serverDigester.addObjectCreate("Server/Listener",

                                   "com.test.server.digester.Listener");

                  serverDigester.addSetProperties("Server/Listener", "className",

                                   "className");

                  serverDigester.addSetProperties("Server/Listener", "debug", "debug");

 

                  // Server对象添加一个Listener对象(无论Server节点下有几个Listener对象,都会添加到Vector)

                  serverDigester.addSetNext("Server/Listener", "addListener",

                                   "com.test.server.digester.Listener");

 

                  // 注册Service对象

                  serverDigester.addObjectCreate("Server/Service",

                                   "com.test.server.digester.Service");

                  serverDigester.addSetProperties("Server/Service", "name", "name");

 

                  // server中添加Service

                  serverDigester.addSetNext("Server/Service", "setService",

                                   "com.test.server.digester.Service");

 

                  // Service中添加Engine,调用当前top objectsetEngine函数,参数个数为0

                  serverDigester.addCallMethod("Server/Service/Engine", "setEngine", 0);

                  serverDigester.addSetProperties("Server/Service/Engine","enginetype","enginetype");

 

                  // 开始解析XML文件

                  Server currServer = (Server) serverDigester.parse(inputXMLStream);

                  if (currServer == null) {

                          System.out.println("currServer==null");

                          return;

                  }

                  Vector listenerVector = currServer.getListeners();

                  System.out.println(listenerVector.size());

 

<span style="font-size

根据提供的引用内容,可以看出问题出在conf/server.xml文件的某个Listener配置上,其可能包含了org.apache.catalina.core.AprLifecycleListener和org.apache.catalina.core.JasperListener这两个类的一个或者两个。而根据引用的错误提示,我们可以看出JasperListener这个类没有被找到,因此可以尝试注释掉server.xml文件的JasperListener配置,然后重启Tomcat服务器,看是否能够解决问题。 以下是具体步骤: . 打开conf/server.xml文件。 2. 找到类似于以下的一段配置: ``` <Listener className="org.apache.catalina.core.AprLifecycleListener" SSLEngine="on" /> <Listener className="org.apache.catalina.core.JasperListener" /> Listener className="org.apache.catalina.core.JreMemoryLeakPreventionListener" /> <Listener className="org.apache.catalina.mbeans.ServerLifecycleListener" /> <Listener className="org.apache.catalina.mbeans.GlobalResourcesLifecycleListener" /> ``` 3. 将其的JasperListener配置注释掉,即在该行前面加上“<!--”并在该行后面加上“-->”,如下所示: ``` <Listener className="org.apache.catalina.core.AprLifecycleListener" SSLEngine="on" /> <!--<Listener className="org.apache.catalina.core.JasperListener" />--> <Listener className="org.apache.catalina.core.JreMemoryLeakPreventionListener" /> <Listener className="org.apache.catalina.mbeans.ServerLifecycleListener" /> <Listener className="org.apache.catalina.mbeans.GlobalResourcesLifecycleListener" /> ``` 4. 保存server.xml文件,并重启Tomcat服务器。 5. 检查是否能够正常启动Tomcat服务器,如果能够正常启动,则说明问题已经解决。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

O溺水的鱼0

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值