今日繼續(xù)學(xué)習(xí)Android中使用Pull的XML解析技術(shù)實(shí)現(xiàn)對(duì)XML文件的解析和創(chuàng)建。由于明天休息,時(shí)間比較充裕,所以我也將昨天未總結(jié)的SAX解析技術(shù)在此做個(gè)總結(jié)。
一、SAX解析技術(shù)
Sax使用的是事件驅(qū)動(dòng)的流式解析技術(shù)。事件驅(qū)動(dòng)的流式解析方式是,從文件的開(kāi)始順序解析到文檔的結(jié)束,不可暫停或倒退。當(dāng)解析到文檔的開(kāi)始或結(jié)束、元素的開(kāi)始或結(jié)束等都會(huì)觸發(fā)一個(gè)事件,我們?cè)谑录幚矸椒ㄖ型瓿蓪?duì)數(shù)據(jù)的操作。由此可見(jiàn),我們需要編寫(xiě)實(shí)現(xiàn)了事件接口的類。
1.創(chuàng)建Android工程(eclipse3.5):
Project name:AndroidXML
BuildTarget:Android2.1
Application name:Android XML 解析技術(shù)
Package name:com.changcheng.androidxml
Create Activity:AndroidXML
Min SDK Version:7
2.需要解析的XML文件:
<?xml version="1.0" encoding="UTF-8"?>
<books>
<book id="23">
<name>C++ Primer 4</name>
<price>78</price>
</book>
<book id="20">
<name>Think in Java</name>
<price>76</price>
</book>
</books>
|
該文件存放于src源碼目錄。
3.XML文件對(duì)應(yīng)的實(shí)體Book:
package com.changcheng.androidxml.entity;
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 + "]";
}
}
|
4.Sax解析XML的事件處理類:
Sax的事件處理類必須實(shí)現(xiàn)ContentHandler接口,但我們?cè)谶@個(gè)例子中不需要使用到ContentHandler接口的所有方法,我們僅需要其中的3個(gè)方法。所以Sax為我們提供了一個(gè)沒(méi)有進(jìn)行任何操作的ContentHandler實(shí)現(xiàn)類DefaultHandler。我們直接繼承DefaultHandler類,并重寫(xiě)我們需要的方法即可。
package com.changcheng.androidxml.xml;
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<Book> books;
private Book book;
private String tagName;
public List<Book> getBooks() {
return books;
}
/**
* 接收文檔的開(kāi)始的通知。
*/
@Override
public void startDocument() throws SAXException {
this.books = new ArrayList<Book>();
}
/**
* 接收字符數(shù)據(jù)的通知。
*/
@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));
}
}
}
/**
* 接收元素開(kāi)始的通知。
* 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;
}
/**
* 接收文檔的結(jié)尾的通知。
* 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;
}
}
|
5.編寫(xiě)測(cè)試Sax解析XML的類
在創(chuàng)建工程時(shí),生成的AndroidXML.java,并沒(méi)有被使用到。因?yàn)槲覀兪褂?span lang="EN-US" xml:lang="EN-US">Android的單元測(cè)試,運(yùn)行上面的程序。
編寫(xiě)Android單元測(cè)試類:
package com.changcheng.androidxml.test;
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";
/**
* 測(cè)試Sax解析XML
* @throws Throwable
*/
public void testAndroidSaxReadXML() throws Throwable{
InputStream file = this.getClass().getClassLoader().getResourceAsStream("books.xml");
try {
List<Book> books = AndoridSaxXml.readXML(file);
Log.i(TAG, books.toString());
} catch (Exception e) {
Log.e(TAG, e.toString());
}
}
}
|
測(cè)試類必須繼承自AndroidTestCase類,Android的單元測(cè)試使用的是JUnit3,所以在我們的測(cè)試方法名稱要以test開(kāi)頭。
再編寫(xiě)一個(gè)AndoridSaxXml(測(cè)試類中使用到的)類:
package com.changcheng.androidxml.xml;
import java.io.InputStream;
import java.util.List;
import javax.xml.parsers.SAXParser;
import javax.xml.parsers.SAXParserFactory;
import com.changcheng.androidxml.entity.Book;
public class AndoridSaxXml {
public static List<Book> readXML(InputStream inputStream) throws Exception {
// 創(chuàng)建Sax解析
SAXParserFactory saxParFac = SAXParserFactory.newInstance();
SAXParser saxParser = saxParFac.newSAXParser();
SaxXmlContentHandler handler = new SaxXmlContentHandler();
// 解析XML文件
saxParser.parse(inputStream, handler);
inputStream.close();
return handler.getBooks();
}
}
|
6.運(yùn)行測(cè)試
在outline面板中的testAndroidSaxReadXML方法或在TestAndroidXML類的testAndroidSaxReadXML方法上右鍵->Debug As->Android Junit Test。運(yùn)行結(jié)束后在LogCat面板中查看運(yùn)行結(jié)束。
關(guān)于使用Sax生成XML文檔,我在此就不做總結(jié)了。下面的Pull技術(shù)才是我們進(jìn)行Android開(kāi)發(fā)的重點(diǎn)。
二、Pull解析技術(shù)
Pull解析技術(shù)與Sax解析技術(shù)原理相同,但比Sax解析簡(jiǎn)單,它們的解析速度和占用的資源差不多。Android內(nèi)部使用的XML解析技術(shù)正是Pull,Android官方推薦開(kāi)發(fā)者們使用Pull解析技術(shù)。Pull解析技術(shù)是第三方開(kāi)發(fā)的開(kāi)源技術(shù),它同樣可以應(yīng)用于JavaSE開(kāi)發(fā)。下面我們使用Pull解析技術(shù)解析XML文件,然后再使用Pull技術(shù)生成XML文件。
Pull解析XML文檔
1.XML文件
依然使用上面的books.xml
2.XML文檔對(duì)應(yīng)的實(shí)體Book
依然使用上面的Book.java
3.Pull解析XML類
package com.changcheng.androidxml.xml;
import java.io.InputStream;
import java.io.Writer;
import java.util.ArrayList;
import java.util.List;
import org.xmlpull.v1.XmlPullParser;
import org.xmlpull.v1.XmlPullParserFactory;
import org.xmlpull.v1.XmlSerializer;
import android.util.Xml;
import com.changcheng.androidxml.entity.Book;
public class AndroidPullXML {
public static List<Book> readXML(InputStream inputStream,
String inputEncoding) throws Exception {
// 創(chuàng)建Pull解析
XmlPullParserFactory pullParserFactory = XmlPullParserFactory
.newInstance();
XmlPullParser pullParser = pullParserFactory.newPullParser();
// 解析XML
pullParser.setInput(inputStream, inputEncoding);
// 開(kāi)始
int eventType = pullParser.getEventType();
List<Book> books = null;
Book book = null;
while (eventType != XmlPullParser.END_DOCUMENT) {
String nodeName = pullParser.getName();
switch (eventType) {
// 文檔開(kāi)始
case XmlPullParser.START_DOCUMENT:
books = new ArrayList<Book>();
break;
// 節(jié)點(diǎn)開(kāi)始
case XmlPullParser.START_TAG:
if ("book".equals(nodeName)) {
book = new Book();
book.setId(Integer
.parseInt(pullParser.getAttributeValue(0)));
} else if ("name".equals(nodeName)) {
book.setName(pullParser.nextText());
} else if ("price".equals(nodeName)) {
book.setPrice(Float.parseFloat(pullParser.nextText()));
}
break;
// 節(jié)點(diǎn)結(jié)束
case XmlPullParser.END_TAG:
if ("book".equals(nodeName)) {
books.add(book);
book = null;
}
break;
}
eventType = pullParser.next();
}
return books;
}
}
|
4.編寫(xiě)測(cè)試Pull解析XML類
在sax測(cè)試類TestAndroidXML中添加一個(gè)測(cè)試方法:
/**
* 測(cè)試Pull解析XML
* @throws Throwable
*/
public void testAndroidPullReadXML() throws Throwable {
InputStream file = this.getClass().getClassLoader().getResourceAsStream("books.xml");
try {
List<Book> books = AndroidPullXML.readXML(file, "UTF-8");
Log.i(TAG, books.toString());
} catch (Exception e) {
Log.e(TAG, e.toString());
}
}
|
5.運(yùn)行測(cè)試
在outline面板中的testAndroidPullReadXML方法或在TestAndroidXML類的testAndroidPullReadXML方法上右鍵->Debug As->Android Junit Test。運(yùn)行結(jié)束后在LogCat面板中查看運(yùn)行結(jié)束。
Pull生成XML文檔
使用Pull生成上面的books.xml文檔。
1.在AndroidPullXML類中添加一個(gè)方法:
public static void writeXML(Writer writer, List<Book> books)
throws Exception {
// 創(chuàng)建XML生成器
XmlSerializer writexml = Xml.newSerializer();
writexml.setOutput(writer);
// 生成XML文檔
writexml.startDocument("UTF-8", true);
writexml.startTag("", "books");
for (Book book : books) {
// name
writexml.startTag("", "name");
writexml.attribute("", "id", book.getId() + "");
writexml.text(book.getName());
writexml.endTag("", "name");
// price
writexml.startTag("", "price");
writexml.text(book.getPrice() + "");
writexml.endTag("", "price");
}
//
writexml.endTag("", "books");
}
|
2.編寫(xiě)測(cè)試Pull生成XML方法
在sax測(cè)試類TestAndroidXML中添加一個(gè)測(cè)試方法:
/**
* 測(cè)試Pull生成XML
* @throws Throwable
*/
public void testAndroidPullWriteXML() throws Throwable {
// 生成到內(nèi)存中。(也可以生成到文件中,那就需要定義一個(gè)文件輸出流。)
StringWriter writer = new StringWriter();
// 添加三本書(shū)
List<Book> books = new ArrayList<Book>();
books.add(new Book(1, "C", 89));
books.add(new Book(1, "C++", 100));
books.add(new Book(1, "Java", 87));
books.add(new Book(1, "JavaEE", 95));
// 生成XML
AndroidPullXML.writeXML(writer, books);
// 打印結(jié)果
Log.i(TAG, books.toString());
}
|
3.運(yùn)行測(cè)試
在outline面板中的testAndroidPullWriteXML方法或在TestAndroidXML類的testAndroidPullWriteXML方法上右鍵->Debug As->Android Junit Test。運(yùn)行結(jié)束后在LogCat面板中查看運(yùn)行結(jié)束。
OK,使用Sax和Pull在Android中解析XML文檔到此完成。在Andorid中還可以使用DOM技術(shù),使用DOM技術(shù)解析在我們學(xué)習(xí)JavaWeb基礎(chǔ)時(shí),已經(jīng)做了總結(jié),在此就不再介紹了。