接下来tomcat要load了,看下面一些程序片段
public void load() {
long t1 = System.nanoTime();
initDirs();
initNaming();
Digester digester = createStartDigester();
........
digester.push(this);
digester.parse(inputSource);
........
getServer().init();
令人费解的如何实例化server的?digester有何作用?原来digester是apache的common项目,作用是讲XML转成Object。tomcat读取配置文件conf\server.xml,实例化server对象。形同:
// Configure the actions we will be using
digester.addObjectCreate("Server",
"org.apache.catalina.core.StandardServer",
"className");
digester.addSetProperties("Server");
digester.addSetNext("Server",
"setServer",
"org.apache.catalina.Server");
要读懂这些还真费解,这里写了个小例子。
首先xml文件为
<?xml version="1.0"?> <catalog library="somewhere"> <book> <author>Author 1</author> <title>Title 1</title> </book> <book> <author>Author 2</author> <title>His One Book</title> </book> <book> <author>Author 3</author> <title>His Other Book</title> </book> </catalog>
Book.java
package com.xiao;
public class Book {
private String author;
private String title;
public Book() {}
public void setAuthor( String rhs ) { author = rhs; }
public void setTitle( String rhs ) { title = rhs; }
public String getAuthor( ) { return author; }
public String getTitle( ) { return title; }
public String toString() {
return "Book: Author='" + author + "' Title='" + title + "'";
}
}
Catalog.java
package com.xiao;
import java.util.Vector;
public class Catalog {
private Vector<Book> books;
public Catalog() {
books = new Vector<Book>();
}
public void addBook( Book rhs ) {
books.addElement( rhs );
}
public String toString() {
String newline = System.getProperty( "line.separator" );
StringBuffer buf = new StringBuffer();
buf.append( "--- Books ---" ).append( newline );
for( int i=0; i<books.size(); i++ ){
Book book = books.elementAt(i);
buf.append( book.toString()).append( newline );
}
return buf.toString();
}
}
CreateCatalog.java
package com.xiao;
import java.io.IOException;
import org.apache.tomcat.util.digester.Digester;
import org.xml.sax.SAXException;
public class CreateCatalog {
protected Catalog ct;
public void SetCatalog(Catalog ol){
ct = ol;
}
public Catalog GetCatalog(){
return this.ct;
}
public String toString() {
return ct.toString();
}
public Digester createStartDigester()throws IOException, SAXException
{
Digester digester = new Digester();
digester.setValidating( false );
//解析XML时,遇到catalog,就实例化一个com.xiao.Catalog对象,并且压栈
digester.addObjectCreate( "catalog", "com.xiao.Catalog");
//对catalog,调用栈的次top对象(现在还没有压入,父对象)的SetCatalog函数。
//passing the element that is on the top of the stack, which must be of type com.xiao.Catalog
//This is the rule that causes the parent/child relationship to be created.
digester.addSetNext("catalog", "SetCatalog", "com.xiao.Catalog");
digester.addObjectCreate( "catalog/book","com.xiao.Book");
//对rule,调用当前top object的setAuthor函数,参数个数为1
digester.addCallMethod("catalog/book/author", "setAuthor",1);
//对rule,添加第一个参数值
digester.addCallParam("catalog/book/author", 0);
digester.addCallMethod("catalog/book/title", "setTitle",1);
digester.addCallParam("catalog/book/title", 0);
//此时次top的object就是com.xiao.Catalog,调用它的addBook函数,将com.xiao.Book传入
digester.addSetNext("catalog/book", "addBook", "com.xiao.Book");
return (digester);
}
}
TestDigester.java
package com.xiao;
import org.apache.tomcat.util.digester.Digester;
import org.xml.sax.InputSource;
import org.xml.sax.SAXException;
import java.io.*;
public class TestDigester {
public static void main(String[] args) throws IOException, SAXException {
// TODO Auto-generated method stub
CreateCatalog cc = new CreateCatalog();
Digester digester = cc.createStartDigester();
String configFile = "xiapingtest/ts.xml";
InputSource inputSource = null;
InputStream inputStream = null;
File file = new File(System.getProperty("user.dir"),configFile);
inputSource = new InputSource("file://" + file.getAbsolutePath());
inputStream = new FileInputStream(file);
inputSource.setByteStream(inputStream);
//在加入CreateCatalog对象,这个是第一个压入的对象
digester.push(cc);
//处理xml文件,逐个加入对象
digester.parse(inputSource);
System.out.println(cc.toString());
}
}
部分解释以及在代码注释里。