今日繼續(xù)學(xué)習(xí)Android中使用Pull的XML解析技術(shù)實現(xiàn)對XML文件的解析和創(chuàng)建。由于明天休息,時間比較充裕,所以我也將昨天未總結(jié)的SAX解析技術(shù)在此做個總結(jié)。
一、SAX解析技術(shù)
Sax使用的是事件驅(qū)動的流式解析技術(shù)。事件驅(qū)動的流式解析方式是,從文件的開始順序解析到文檔的結(jié)束,不可暫停或倒退。當解析到文檔的開始或結(jié)束、元素的開始或結(jié)束等都會觸發(fā)一個事件,我們在事件處理方法中完成對數(shù)據(jù)的操作。由此可見,我們需要編寫實現(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文件對應(yīng)的實體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的事件處理類必須實現(xiàn)ContentHandler接口,但我們在這個例子中不需要使用到ContentHandler接口的所有方法,我們僅需要其中的3個方法。所以Sax為我們提供了一個沒有進行任何操作的ContentHandler實現(xiàn)類DefaultHandler。我們直接繼承DefaultHandler類,并重寫我們需要的方法即可。
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;
}
/**
* 接收文檔的開始的通知。
*/
@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));
}
}
}
/**
* 接收元素開始的通知。
* 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.編寫測試Sax解析XML的類
在創(chuàng)建工程時,生成的AndroidXML.java,并沒有被使用到。因為我們使用Android的單元測試,運行上面的程序。
編寫Android單元測試類:
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";
/**
* 測試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());
}
}
}
|
測試類必須繼承自AndroidTestCase類,Android的單元測試使用的是JUnit3,所以在我們的測試方法名稱要以test開頭。
再編寫一個AndoridSaxXml(測試類中使用到的)類:
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.運行測試
在outline面板中的testAndroidSaxReadXML方法或在TestAndroidXML類的testAndroidSaxReadXML方法上右鍵->Debug As->Android Junit Test。運行結(jié)束后在LogCat面板中查看運行結(jié)束。
關(guān)于使用Sax生成XML文檔,我在此就不做總結(jié)了。下面的Pull技術(shù)才是我們進行Android開發(fā)的重點。
二、Pull解析技術(shù)
Pull解析技術(shù)與Sax解析技術(shù)原理相同,但比Sax解析簡單,它們的解析速度和占用的資源差不多。Android內(nèi)部使用的XML解析技術(shù)正是Pull,Android官方推薦開發(fā)者們使用Pull解析技術(shù)。Pull解析技術(shù)是第三方開發(fā)的開源技術(shù),它同樣可以應(yīng)用于JavaSE開發(fā)。下面我們使用Pull解析技術(shù)解析XML文件,然后再使用Pull技術(shù)生成XML文件。
Pull解析XML文檔
1.XML文件
依然使用上面的books.xml
2.XML文檔對應(yīng)的實體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);
// 開始
int eventType = pullParser.getEventType();
List<Book> books = null;
Book book = null;
while (eventType != XmlPullParser.END_DOCUMENT) {
String nodeName = pullParser.getName();
switch (eventType) {
// 文檔開始
case XmlPullParser.START_DOCUMENT:
books = new ArrayList<Book>();
break;
// 節(jié)點開始
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é)點結(jié)束
case XmlPullParser.END_TAG:
if ("book".equals(nodeName)) {
books.add(book);
book = null;
}
break;
}
eventType = pullParser.next();
}
return books;
}
}
|
4.編寫測試Pull解析XML類
在sax測試類TestAndroidXML中添加一個測試方法:
/**
* 測試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.運行測試
在outline面板中的testAndroidPullReadXML方法或在TestAndroidXML類的testAndroidPullReadXML方法上右鍵->Debug As->Android Junit Test。運行結(jié)束后在LogCat面板中查看運行結(jié)束。
Pull生成XML文檔
使用Pull生成上面的books.xml文檔。
1.在AndroidPullXML類中添加一個方法:
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.編寫測試Pull生成XML方法
在sax測試類TestAndroidXML中添加一個測試方法:
/**
* 測試Pull生成XML
* @throws Throwable
*/
public void testAndroidPullWriteXML() throws Throwable {
// 生成到內(nèi)存中。(也可以生成到文件中,那就需要定義一個文件輸出流。)
StringWriter writer = new StringWriter();
// 添加三本書
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.運行測試
在outline面板中的testAndroidPullWriteXML方法或在TestAndroidXML類的testAndroidPullWriteXML方法上右鍵->Debug As->Android Junit Test。運行結(jié)束后在LogCat面板中查看運行結(jié)束。
OK,使用Sax和Pull在Android中解析XML文檔到此完成。在Andorid中還可以使用DOM技術(shù),使用DOM技術(shù)解析在我們學(xué)習(xí)JavaWeb基礎(chǔ)時,已經(jīng)做了總結(jié),在此就不再介紹了。