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;
}
@Override
public 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;
}
/**接收文档的开始的通知。*/
@Override
public void startDocument() throws SAXException {
this.books = new ArrayList();
}
/**接收字符数据的通知。*/
@Override
public 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:元素的属性集合
*/
@Override
public 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:元素的限定名(带前缀)
*/
@Override
public 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面板中查看运行结束。