Android XML讀寫(五)
de >RssHandlerde > 類擴(kuò)展了 de >org.xml.sax.helpers.DefaultHandlerde > 類。該類為 SAX 解析器生成的事件所對(duì)應(yīng)的所有方法都提供了一個(gè)默認(rèn)的非操作實(shí)現(xiàn)。這允許子類根據(jù)需要僅覆蓋一些方法。de >RssHandlerde > 提供了一個(gè)額外的 API,即 de >getMessagesde >。它返回處理程序在從 SAX 解析器接收事件時(shí)所收集的 de >Messagede > 對(duì)象列表。它有另外兩個(gè)內(nèi)部變量,de >currentMessagede > 針對(duì)被解析的 de >Messagede > 實(shí)例,以及名稱為 de >builderde > 的 de >StringBuilderde > 變量,用于存儲(chǔ)文本節(jié)點(diǎn)中的字符數(shù)據(jù)。解析器將相應(yīng)事件發(fā)送給處理程序時(shí)會(huì)調(diào)用 de >startDocumentde > 方法,這兩個(gè)變量的初始化操作就是在此時(shí)完成。
查看de >startElementde > 方法。在XML文檔中每次遇到開始標(biāo)記時(shí)都會(huì)調(diào)用它。您只關(guān)心該標(biāo)記何時(shí)為 de >ITEMde > 標(biāo)記。對(duì)于這種情況,您將創(chuàng)建一個(gè)新的 de >Messagede >?,F(xiàn)在來看 de >charactersde > 方法。遇到文本節(jié)點(diǎn)中的字符數(shù)據(jù)時(shí)便會(huì)調(diào)用此方法。數(shù)據(jù)只是被添加到 de >builderde > 變量中。***,我們來看 de >endElementde > 方法。遇到結(jié)束標(biāo)記時(shí)會(huì)調(diào)用此方法。對(duì)于與某 de >Messagede > 屬性相對(duì)應(yīng)的標(biāo)記,如 de >TITLEde > 和 de >LINKde >,則使用 de >builderde > 變量中的數(shù)據(jù)在 de >currentMessagede > 上設(shè)置適當(dāng)?shù)膶傩浴H绻Y(jié)束標(biāo)記是一個(gè) de >ITEMde >,則 de >currentMessagede > 將被添加到 Messages 列表中。所有這些都是非常典型的 SAX 解析;此處的一切都不是 Android 所特有的。因此,如果您知道如何編寫 Java SAX 解析器,則應(yīng)該知道如何編寫 Android SAX 解析器。但是,android sdk 確實(shí)在 SAX 上添加了一些便捷的特性。
Android SAX 解析器
java代碼:
- public class AndroidSaxFeedParser extends BaseFeedParser {
- public AndroidSaxFeedParser(String feedUrl) {
- super(feedUrl);
- }
- public List< Message > parse() {
- RssHandler handler = new RssHandler();
- try {
- Xml.parse(this.getInputStream(), Xml.Encoding.UTF_8, handler);
- } catch (Exception e) {
- throw new RuntimeException(e);
- }
- return handler.getMessages();
- }
- }
注意,這個(gè)類仍然使用了一個(gè)標(biāo)準(zhǔn)的 SAX 處理程序,因此您僅僅重用了所示的 de >RssHandlerde >。能夠重用 SAX 處理程序是非常不錯(cuò)的,但其代碼稍微有些復(fù)雜。您可以想像,如果需要解析一個(gè)更加復(fù)雜的 XML 文檔,則處理程序可能會(huì)帶來各種各樣的 bug。舉例來說,回頭看看 de >endElementde > 方法。注意,在嘗試設(shè)置屬性之前,它檢查了 de >currentMessagede > 是否為 null?,F(xiàn)在,再回頭看看示例 XML。 注意,de >ITEMde > 標(biāo)記外部有一些 de >TITLEde > 和 de >LINKde > 標(biāo)記。這就是使用 null 檢查的原因。否則,每一個(gè) de >TITLEde > 標(biāo)記 會(huì)導(dǎo)致一個(gè) de >NullPointerExceptionde >。Android 提供了自己獨(dú)有的 SAX API,它排除了您編寫自己的 SAX 處理程序的需要。
經(jīng)過簡化的 Android SAX 解析器
java代碼:
- public class AndroidSaxFeedParser extends BaseFeedParser {
- public AndroidSaxFeedParser(String feedUrl) {
- super(feedUrl);
- }
- public List< Message > parse() {
- final Message currentMessage = new Message();
- RootElement root = new RootElement("rss");
- final List< Message > messages = new ArrayList< Message >();
- Element channel = root.getChild("channel");
- Element item = channel.getChild(ITEM);
- item.setEndElementListener(new EndElementListener(){
- public void end() {
- messages.add(currentMessage.copy());
- }
- });
- item.getChild(TITLE).setEndTextElementListener(new EndTextElementListener(){
- public void end(String body) {
- currentMessage.setTitle(body);
- }
- });
- item.getChild(LINK).setEndTextElementListener(new EndTextElementListener(){
- public void end(String body) {
- currentMessage.setLink(body);
- }
- });
- item.getChild(DESCRIPTION).setEndTextElementListener(new
- EndTextElementListener(){
- public void end(String body) {
- currentMessage.setDescription(body);
- }
- });
- item.getChild(PUB_DATE).setEndTextElementListener(new EndTextElementListener(){
- public void end(String body) {
- currentMessage.setDate(body);
- }
- });
- try {
- Xml.parse(this.getInputStream(), Xml.Encoding.UTF_8,root.getContentHandler());
- } catch (Exception e) {
- throw new RuntimeException(e);
- }
- return messages;
- }
- }