ContentHandler實(shí)際上是一個(gè)接口,當(dāng)處理特定的XML文件的時(shí)候,就需要為其創(chuàng)建一個(gè)實(shí)現(xiàn)了ContentHandler的類來處理特定的事件,可以說,這個(gè)實(shí)際上就是SAX處理XML文件的核心。下面我們來看看定義在其中的一些方法:
void characters(char[] ch, int start, int length):
這個(gè)方法用來處理在XML文件中讀到字符串,它的參數(shù)是一個(gè)字符數(shù)組,以及讀到的這個(gè)字符串在這個(gè)數(shù)組中的起始位置和長度,我們可以很容易的用String類的一個(gè)構(gòu)造方法來獲得這個(gè)字符串的String類:String charEncontered=new String(ch,start,length)。
void startDocument():
當(dāng)遇到文檔的開頭的時(shí)候,調(diào)用這個(gè)方法,可以在其中做一些預(yù)處理的工作。
void endDocument():
和上面的方法相對應(yīng),當(dāng)文檔結(jié)束的時(shí)候,調(diào)用這個(gè)方法,可以在其中做一些善后的工作。
void startElement(java.lang.String namespaceURI, java.lang.String localName, java.lang.String qName, Attributes atts)
當(dāng)讀到一個(gè)開始標(biāo)簽的時(shí)候,會(huì)觸發(fā)這個(gè)方法。在SAX1.0版本中并不支持名域,而在新的2.0版本中提供了對名域的支持,這兒參數(shù)中的namespaceURI就是名域,localName是標(biāo)簽名,qName是標(biāo)簽的修飾前綴,當(dāng)沒有使用名域的時(shí)候,這兩個(gè)參數(shù)都未null。而atts是這個(gè)標(biāo)簽所包含的屬性列表。通過atts,可以得到所有的屬性名和相應(yīng)的值。要注意的是SAX中一個(gè)重要的特點(diǎn)就是它的流式處理,在遇到一個(gè)標(biāo)簽的時(shí)候,它并不會(huì)紀(jì)錄下以前所碰到的標(biāo)簽,也就是說,在startElement()方法中,所有你所知道的信息,就是標(biāo)簽的名字和屬性,至于標(biāo)簽的嵌套結(jié)構(gòu),上層標(biāo)簽的名字,是否有子元屬等等其它與結(jié)構(gòu)相關(guān)的信息,都是不得而知的,都需要你的程序來完成。這使得SAX在編程處理上沒有DOM來得那么方便。
void endElement(java.lang.String namespaceURI, java.lang.String localName, java.lang.String qName)
這個(gè)方法和上面的方法相對應(yīng),在遇到結(jié)束標(biāo)簽的時(shí)候,調(diào)用這個(gè)方法。
因?yàn)镃ontentHandler是一個(gè)接口,在使用的時(shí)候可能會(huì)有些不方便,因而,SAX中還為其制定了一個(gè)Helper類:DefaultHandler,它實(shí)現(xiàn)了這個(gè)接口,但是其所有的方法體都為空,在實(shí)現(xiàn)的時(shí)候,你只需要繼承這個(gè)類,然后重載相應(yīng)的方法即可。
到這兒SAX的基本知識(shí)已經(jīng)差不多講完了,下面我們來看看兩個(gè)具體的例子,以更好的理解SAX地用法。
SAX編程實(shí)例
我們還是沿用講DOM的時(shí)候使用的那個(gè)文檔例子,但首先,我們先看一個(gè)簡單一些的應(yīng)用,我們希望能夠統(tǒng)計(jì)一下XML文件中各個(gè)標(biāo)簽出現(xiàn)的次數(shù)。這個(gè)例子很簡單,但是足以闡述SAX編程的基本思路了。
一開始當(dāng)然還是import語句了:
import org.xml.sax.helpers.DefaultHandler;
import javax.xml.parsers.*;
import org.xml.sax.*;
import org.xml.sax.helpers.*;
import java.util.*;
import java.io.*;
然后,我們創(chuàng)建一個(gè)繼承于DefaultHandler的類,具體的程序邏輯在這兒可以暫且放在一邊,要注意的是程序的結(jié)構(gòu):
public class SAXCounter extends DefaultHandler {
private Hashtable tags; //這個(gè)Hashtable用來記錄tag出現(xiàn)的次數(shù)
// 處理文檔前的工作
public void startDocument() throws SAXException {
tags = new Hashtable();//初始化Hashtable
}
//對每一個(gè)開始元屬進(jìn)行處理
public void startElement(String namespaceURI, String localName,
String rawName, Attributes atts)
throws SAXException
{
String key = localName;
Object value = tags.get(key);
if (value == null) {
// 如果是新碰到的標(biāo)簽,這在Hastable中添加一條記錄
tags.put(key, new Integer(1));
} else {
// 如果以前碰到過,得到其計(jì)數(shù)值,并加1
int count = ((Integer)value).intValue();
count++;
tags.put(key, new Integer(count));
}
}
//解析完成后的統(tǒng)計(jì)工作
相關(guān)推薦:計(jì)算機(jī)等級(jí)考試二級(jí)Java經(jīng)典算法大全匯總
北京 | 天津 | 上海 | 江蘇 | 山東 |
安徽 | 浙江 | 江西 | 福建 | 深圳 |
廣東 | 河北 | 湖南 | 廣西 | 河南 |
海南 | 湖北 | 四川 | 重慶 | 云南 |
貴州 | 西藏 | 新疆 | 陜西 | 山西 |
寧夏 | 甘肅 | 青海 | 遼寧 | 吉林 |
黑龍江 | 內(nèi)蒙古 |