Android 采用PULL來解析XML

jopen 12年前發布 | 117K 次閱讀 Android Android開發 移動開發

概述:

PULL解析器小巧輕便,解析速度快,簡單易用,非常適合在Android移動設備中使用,Android系統內部在解析各種XML時也是用PULL解析器,Android官方推薦開發者們使用Pull解析技術。Pull解析技術是第三方開發的開源技術,它同樣可以應用于JavaSE開發。

PULL 的工作原理:

XML pull提供了開始元素和結束元素。當某個元素開始時,可以調用parsernextTextXML文檔中提取所有字符數據。當解析到一個文檔結束時,自動生成EndDocument事件。

常用的XML pull的接口和類:

XmlPullParser該解析器是一個在org.xmlpull.v1中定義的解析功能的接口。

XmlSerializer它是一個接口,定義了XML信息集的序列。

XmlPullParserFactory這個類用于在XMPULL V1 API中創建XML Pull解析器。

XmlPullParserException拋出單一的XML pull解析器相關的錯誤。

PULL解析器的運行方式和SAX類似,都是基于事件的模式。

不同的是,在PULL解析過程中返回的是數字,且我們需要自己獲取產生的事件然后做相應的操作,而不像SAX那樣由處理器觸發一種事件的方法,執行我們的代碼:

讀取到xml聲明返回 START_DOCUMENT; 結束返回 END_DOCUMENT ; 開始標簽返回 START_TAG;

結束標簽返回 END_TAG; 文本返回 TEXT。

幾種解析技術的比較與總結: 
對于Android的移動設備而言,因為設備的資源比較寶貴,內存是有限的,所以我們需要選擇適合的技術來解析XML,這樣有利于提高訪問的速度。

DOM在處理XML文件時,將XML文件解析成樹狀結構并放入內存中進行處理。當XML文件較小時,我們可以選DOM,因為它簡單、直觀。

SAX則是以事件作為解析XML文件的模式,它將XML文件轉化成一系列的事件,由不同的事件處理器來決定如何處理。XML文件較大時,選擇SAX技術是比較合理的。雖然代碼量有些大,但是它不需要將所有的XML文件加載到內存中。這樣對于有限的Android內存更有效,而且Android提供了一種傳統的SAX使用方法以及一個便捷的SAX包裝器。 使用AndroidutilXml類,從示例中可以看出,會比使用 SAX來得簡單。

XML pull解析并未像SAX解析那樣監聽元素的結束,而是在開始處完成了大部分處理。這有利于提早讀取XML文件,可以極大的減少解析時間,這種優化對于連接速度較漫的移動設備而言尤為重要。對于XML文檔較大但只需要文檔的一部分時,XML Pull解析器則是更為有效的方法。


開發實例:

在asset定義了books.xml文件,對于它,只要是一份格式良好的xml文檔即可,具體內容根據需要定義。

讀取XML:從定義好的books.xml文件解析并顯示出來;

寫入XML:將解析后的內容再寫入到本地,這里也保存為名字為books.xml的文件。



例如:books.xml定義如下形式:

<?xml version="1.0" encoding="utf-8"?>
<books>
    <book>
        <id>1001</id>
        <name>Thinking In Java</name>
        <price>80.00</price>
    </book>
    <book>
        <id>1002</id>
        <name>Core Java</name>
        <price>90.00</price>
    </book>
    <book>
        <id>1003</id>
        <name>Hello, Andriod</name>
        <price>100.00</price>
    </book>
</books>  


完整代碼:

PULLParserActivity:

package com.xsjayz.xml;

import java.io.FileOutputStream;
import java.io.InputStream;
import java.util.List;

import android.app.Activity;
import android.content.Context;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.Button;
import android.widget.TextView;

/**
 * PULL解析器,這里只是簡單的定義了一個TextView顯示解析結果,兩個Button操作。
 * 
 * @since 2012-08-23
 */
public class PULLParserActivity extends Activity {

    private static final String TAG = "XML";
    private TextView textView = null;
    private Button readBtn = null;
    private Button writeBtn = null;

    private BookParser parser;
    private List<Book> booksList;

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);

        textView = (TextView) findViewById(R.id.txt);
        readBtn = (Button) findViewById(R.id.read_btn);
        writeBtn = (Button) findViewById(R.id.write_btn);

        readBtn.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                try {
                    InputStream is = getAssets().open("books.xml");
                    parser = new PULLBookParser();
                    booksList = parser.parse(is);

                    for (Book book : booksList) {
                        Log.i(TAG, book.toString());
                    }

                    for (Book book : booksList) {
                        textView.setText(textView.getText() + book.toString());
                    }
                } catch (Exception e) {
                    Log.e(TAG, e.getMessage());
                }
            }
        });

        writeBtn.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                try {
                    String xmlString = parser.serialize(booksList); // 序列化
                    FileOutputStream fos = openFileOutput("books.xml",
                            Context.MODE_PRIVATE);
                    fos.write(xmlString.getBytes("UTF-8"));
                } catch (Exception e) {
                    Log.e(TAG, e.getMessage());
                }
            }
        });
    }
}

PULLBookParser:

package com.xsjayz.xml;

import java.io.InputStream;
import java.io.StringWriter;
import java.util.ArrayList;
import java.util.List;

import org.xmlpull.v1.XmlPullParser;
import org.xmlpull.v1.XmlSerializer;

import android.util.Xml;

/**
 * PULL解析器,實現了BookParser接口
 */
public class PULLBookParser implements BookParser {

    /**
     * @param is
     * @return booksList
     */
    @Override
    public List<Book> parse(InputStream is) throws Exception {

        List<Book> booksList = null;
        Book book = null;

        // 由android.util.Xml創建一個XmlPullParser實例
        XmlPullParser parser = Xml.newPullParser();
        // 設置輸入流 并指明編碼方式
        parser.setInput(is, "UTF-8");
        // 產生第一個事件
        int eventType = parser.getEventType();

        while (eventType != XmlPullParser.END_DOCUMENT) {

            switch (eventType) {
            // 判斷當前事件是否為文檔開始事件
            case XmlPullParser.START_DOCUMENT:
                booksList = new ArrayList<Book>(); // 初始化books集合
                break;

            // 判斷當前事件是否為標簽元素開始事件
            case XmlPullParser.START_TAG:

                if (parser.getName().equals("book")) { // 判斷開始標簽元素是否是book
                    book = new Book();
                } else if (parser.getName().equals("id")) {
                    eventType = parser.next();
                    // 得到book標簽的屬性值,并設置book的id
                    book.setId(Integer.parseInt(parser.getText()));
                } else if (parser.getName().equals("name")) { // 判斷開始標簽元素是否是book
                    eventType = parser.next();
                    book.setName(parser.getText());
                } else if (parser.getName().equals("price")) { // 判斷開始標簽元素是否是price
                    eventType = parser.next();
                    book.setPrice(Float.parseFloat(parser.getText()));
                }
                break;

            // 判斷當前事件是否為標簽元素結束事件
            case XmlPullParser.END_TAG:
                if (parser.getName().equals("book")) { // 判斷結束標簽元素是否是book
                    booksList.add(book); // 將book添加到books集合
                    book = null;
                }
                break;
            }
            // 進入下一個元素并觸發相應事件
            eventType = parser.next();
        }
        return booksList;
    }

    /**
     * @param books
     * @return writer.toString()
     */
    @Override
    public String serialize(List<Book> books) throws Exception {

        // 由android.util.Xml創建一個XmlSerializer實例
        XmlSerializer serializer = Xml.newSerializer();
        StringWriter writer = new StringWriter();
        // 設置輸出方向為writer
        serializer.setOutput(writer);
        serializer.startDocument("UTF-8", true);
        serializer.startTag("", "books");

        for (Book book : books) {
            serializer.startTag("", "book");
            serializer.attribute("", "id", book.getId() + "");

            serializer.startTag("", "name");
            serializer.text(book.getName());
            serializer.endTag("", "name");

            serializer.startTag("", "price");
            serializer.text(book.getPrice() + "");
            serializer.endTag("", "price");

            serializer.endTag("", "book");
        }
        serializer.endTag("", "books");
        serializer.endDocument();

        return writer.toString();
    }
}

BookParser:

package com.xsjayz.xml;

import java.io.InputStream;
import java.util.List;

public interface BookParser {

    /**
     * 解析輸入流 得到Book對象集合
     * 
     * @param is
     * @throws Exception
     */
    public List<Book> parse(InputStream is) throws Exception;

    /**
     * 序列化Book對象集合 得到XML形式的字符串
     * 
     * @param books
     * @throws Exception
     */
    public String serialize(List<Book> books) throws Exception;
}

Book:

package com.xsjayz.xml;

/**
 * Book的model類,定義了Book的所有狀態,訪問器和修改器,重寫了toString方法
 */
public class Book {

    private int id;
    private String name;
    private float 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 "id:" + id + "\nname:" + name + "\nprice:" + price + "\n";
    }
}

 本文由用戶 jopen 自行上傳分享,僅供網友學習交流。所有權歸原作者,若您的權利被侵害,請聯系管理員。
 轉載本站原創文章,請注明出處,并保留原始鏈接、圖片水印。
 本站是一個以用戶分享為主的開源技術平臺,歡迎各類分享!