在Android中解析XML文主要有三種方式,分別爲Simple API for XML(SAX)、Document Object Model(DOM)和Android附帶的PULL解析器。其中SAX是一個解析速度非常快並且佔用內存少的XML解析器,非常適合Android手機等移動設備。
SAX解析XML文件採用事件驅動的方式進行,也就是說,SAX是逐行掃描文件,遇到符合條件的設定條件後就會觸發特定的事件,回調你寫好的事件處理程序。使用SAX的優勢在於其解析速度較快,佔用內存較少(相對於DOM而言)。而且SAX在解析文件的過程中得到自己需要的信息後可以隨時終止解析,並不一定要等文件全部解析完畢。凡事有利必有弊,其劣勢在於SAX採用的是流式處理方式,當遇到某個標籤的時候,它並不會記錄下以前所遇到的標籤,也就是說,在處理某個標籤的時候,比如在startElement方法中,所能夠得到的信息就是標籤的名字和屬性,至於標籤內部的嵌套結構,上層標籤、下層標籤以及其兄弟節點的名稱等等與其結構相關的信息都是不得而知的。實際上就是把XML文件的結構信息丟掉了,如果需要得到這些信息的話,只能你自己在程序裏進行處理了。所以相對DOM而言,SAX處理XML文檔沒有DOM方便,SAX處理的過程相對DOM而言也比較複雜。
使用SAX解析XML文件一般有以下五個步驟:
1、創建一個SAXParserFactory對象(通過類名很容易得知它利用工廠方法模式實現的);
2、調用SAXParserFactory中的newSAXParser方法創建一個SAXParser對象;
3、然後在調用SAXParser中的getXMLReader方法獲取一個XMLReader對象;
4、在XMLReader中註冊事件處理接口,一般有ContentHandler、ErrorHandler、DTDHandler、EntityHandler四種;
5、調用XMLReader中的parse方法解析指定的XML字符串對象;
步驟四中提到的四個Handler是事件處理接口,SAX的事件處理函數就定義在這四個接口中,利用SAX解析XML文件需要重寫接口中的方法。其中ContentHandler用來處理XML中的內容,ErrorHandler用來處理錯誤,DTDHandler用來處理DTD,EntityHandler用來處理XML文檔中的實體;最常用的是ContentHandler這個接口,下面是該接口中的一些常用方法:
startDocument()
當遇到文檔的開頭的時候,調用這個方法,可以在其中做一些預處理的工作。
endDocument()
和上面的方法相對應,當文檔結束的時候,調用這個方法,可以在其中做一些善後的工作。
startElement(String uri, String localName, String qName, Attributes atts)
當讀到一個開始標籤的時候,會觸發這個方法。uri是命名空間(通過xmlns聲明),localName是不帶命名空間前綴的標籤名,qName是帶命名空間前綴的標籤名。通過atts可以得到所有的屬性名和相應的值。注意,如果沒有指定Namespace,則qName可能爲空,當然不同的SAX實現會有所不同,比如在Android中qName爲空,而J2SE中localName爲空,所以想要總是得到標籤名,就需要檢查這兩個參數的值了。
endElement(String uri, String localName, String name)
這個方法和上面的方法相對應,在遇到結束標籤的時候,調用這個方法。
characters(char[] ch, int start, int length)
這個方法用來處理在XML文件中讀到的內容,第一個參數爲文件的字符串內容,後面兩個參數是讀到的字符串在這個數組中的起始位置和長度,使用new String(ch,start,length)就可以獲取內容。
下面是利用SAX解析XML文件的DEMO程序,首先定義一個XML文件users.xml,內容如下:
<?xml version="1.0" encoding="utf-8"?>
<users>
<user id="1">
<name>wuxianglong</name>
<password>199098</password>
</user>
<long:user id="2">
<name>wuwenyuan</name>
<password>199189</password>
</long:user>
</users>
其中第二個user標籤中字符串long就是所謂的前綴,標籤中的id則是該標籤的一個屬性。然後我們來實現ContentHandler這個接口,並重寫其中的方法來處理上面的XML文件,代碼如下:
package sax.test;
import java.util.ArrayList;
import org.xml.sax.Attributes;
import org.xml.sax.SAXException;
import org.xml.sax.helpers.DefaultHandler;
public class MyHandler extends DefaultHandler {
private ArrayList<User> users;
private User user;
private String content;
public ArrayList<User> getUsers() {
return users;
}
@Override
public void startDocument() throws SAXException {
super.startDocument();
users = new ArrayList<User>();
System.out.println("----------Start Parse Document----------" );
}
@Override
public void endDocument() throws SAXException {
System.out.println("----------End Parse Document----------" );
}
@Override
public void characters(char[] ch, int start, int length)
throws SAXException {
super.characters(ch, start, length);
// 獲得標籤中的文本
content = new String(ch, start, length);
}
@Override
public void startElement(String uri, String localName, String qName,
Attributes attributes) throws SAXException {
super.startElement(uri, localName, qName, attributes);
// 打印出localname和qName
System.out.println("LocalName->" + localName);
System.out.println("QName->" + qName);
if ("user".equals(localName)) {
user = new User();
user.setId(Integer.parseInt(attributes.getValue("id")));
}
}
@Override
public void endElement(String uri, String localName, String qName)
throws SAXException {
super.endElement(uri, localName, qName);
if ("name".equals(localName)) {
user.setName(content);
} else if ("password".equals(localName)) {
user.setPassword(content);
} else if ("user".equals(localName)) {
users.add(user);
}
}
}
最後我們編輯TestSAX這個類,獲取users.xml這個文件中的內容,並利用SAX進行解析,代碼如下:package sax.test;
import java.io.IOException;
import javax.xml.parsers.ParserConfigurationException;
import javax.xml.parsers.SAXParser;
import javax.xml.parsers.SAXParserFactory;
import org.xml.sax.InputSource;
import org.xml.sax.SAXException;
import org.xml.sax.XMLReader;
import android.app.Activity;
import android.os.Bundle;
import android.widget.TextView;
public class TestSAX extends Activity {
private TextView text;
/** Called when the activity is first created. */
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
text = (TextView) findViewById(R.id.result);
try {
// 創建一個工廠對象
SAXParserFactory factory = SAXParserFactory.newInstance();
// 通過工廠對象得到一個解析器對象
SAXParser parser = factory.newSAXParser();
// 通過parser得到XMLReader對象
XMLReader reader = parser.getXMLReader();
// 爲reader對象註冊事件處理接口
MyHandler handler = new MyHandler();
reader.setContentHandler(handler);
// 解析指定XML字符串對象
reader.parse(new InputSource(TestSAX.class.getClassLoader().getResourceAsStream("users.xml")));
String usersInfo = "";
for (User user : handler.getUsers()) {
usersInfo += "ID->" + user.getId() + "\n";
usersInfo += "NAME->" + user.getName() + "\n";
usersInfo += "PASSWORD->" + user.getPassword() + "\n";
usersInfo += "\n\n";
}
text.setText(usersInfo);
} catch (SAXException e) {
e.printStackTrace();
} catch (ParserConfigurationException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
}
}
這樣就算是完成了,運行之後得到以下結果:
附帶貼上工程目錄結構,如下圖:
THE END!