Android之SAX解析XML

来源:互联网 发布:在家上学 知乎 编辑:程序博客网 时间:2024/04/29 02:26

Sax使用的是事件驱动的流式解析技术。事件驱动的流式解析方式是,从文件的开始顺序解析到文档的结束,不可暂停或倒退。当解析到文档的开始或结束、元素的开始或结束等都会触发一个事件,我们在事件处理方法中完成对数据的操作。由此可见,我们需要编写实现了事件接口的类。

 1.XML文件对应的实体Book:

package eoe.androidxml;public class Book {private int id;private String name;private float price;public Book() {}public Book(int id, String name, float price) {this.id = id;this.name = name;this.price = price;}public int getId() {return id;}public void setId(int id) {this.id = id;}public String getName() {return name;}public void setName(String name) {this.name = name;}public float getPrice() {return price;}public void setPrice(float price) {this.price = price;}@Overridepublic String toString() {return "Book [name=" + name + ", price=" + price + "]";}}

2.Sax解析XML的事件处理类:

       Sax的事件处理类必须实现ContentHandler接口,但我们在这个例子中不需要使用到ContentHandler接口的所有方法,我们仅需要其中的3个方法。所以Sax为我们提供了一个没有进行任何操作的ContentHandler实现类DefaultHandler。我们直接继承DefaultHandler类,并重写我们需要的方法即可。

package eoe.androidxml.;import java.util.ArrayList;import java.util.List;import org.xml.sax.Attributes;import org.xml.sax.SAXException;import org.xml.sax.helpers.DefaultHandler;import com.changcheng.androidxml.entity.Book;public class SaxXmlContentHandler extends DefaultHandler {private List books;private Book book;private String tagName;public List getBooks() {return books;}/**接收文档的开始的通知。*/@Overridepublic void startDocument() throws SAXException {this.books = new ArrayList();}/**接收字符数据的通知。*/@Overridepublic void characters(char[] ch, int start, int length)throws SAXException {if (this.tagName != null) {String data = new String(ch, start, length);if (this.tagName.equals("name")) {this.book.setName(data);} else if (this.tagName.equals("price")) {this.book.setPrice(Float.parseFloat(data));}}}/*** 接收元素开始的通知。* namespaceURI:元素的命名空间* localName:元素的本地名称(不带前缀)* qName:元素的限定名(带前缀)* atts:元素的属性集合*/@Overridepublic void startElement(String uri, String localName, String qName,Attributes attributes) throws SAXException {if (localName.equals("book")) {book = new Book();book.setId(Integer.parseInt(attributes.getValue(0)));}this.tagName = localName;}/*** 接收文档的结尾的通知。* uri:元素的命名空间* localName:元素的本地名称(不带前缀)* name:元素的限定名(带前缀)*/@Overridepublic void endElement(String uri, String localName, String qName)throws SAXException {if (localName.equals("book")) {this.books.add(this.book);}this.tagName = null;}}

   3.编写测试Sax解析XML的类

       在创建工程时,生成的AndroidXML.java,并没有被使用到。因为我们使用Android的单元测试,运行上面的程序。

       编写Android单元测试类:

package eoe.androidxml;import java.io.InputStream;import java.io.StringWriter;import java.util.ArrayList;import java.util.List;import com.changcheng.androidxml.entity.Book;import com.changcheng.androidxml.xml.AndoridSaxXml;import com.changcheng.androidxml.xml.AndroidPullXML;import android.test.AndroidTestCase;import android.util.Log;public class TestAndroidXML extends AndroidTestCase {private static final String TAG = "TestAndroidXML";/*** 测试Sax解析XML* @throws Throwable*/public void testAndroidSaxReadXML() throws Throwable{InputStream file = this.getClass().getClassLoader().getResourceAsStream("books.xml");try {List books = AndoridSaxXml.readXML(file);Log.i(TAG, books.toString());} catch (Exception e) {Log.e(TAG, e.toString());}}}

4.运行测试

       在outline面板中的testAndroidSaxReadXML方法或在TestAndroidXML类的testAndroidSaxReadXML方法上右键->Debug As->Android Junit Test。运行结束后在LogCat面板中查看运行结束。
关于使用Sax生成XML文档,我在此就不做总结了。

原创粉丝点击