Android之SAX解析XML

来源:http://www.cnblogs.com/zhangmiao14/archive/2016/12/15/6183505.html
-Advertisement-
Play Games

一.SAX解析方法介紹 SAX(Simple API for XML)是一個解析速度快並且占用記憶體少的XML解析器,非常適合用於Android等移動設備。 SAX解析器是一種基於事件的解析器,事件驅動的流式解析方式是,從文件的開始順序解析到文檔的結束,不可暫停或倒退。它的核心是事件處理模式,主要是圍 ...


一.SAX解析方法介紹

  SAX(Simple API for XML)是一個解析速度快並且占用記憶體少的XML解析器,非常適合用於Android等移動設備。

  SAX解析器是一種基於事件的解析器,事件驅動的流式解析方式是,從文件的開始順序解析到文檔的結束,不可暫停或倒退。它的核心是事件處理模式,主要是圍繞著事件源以及事件處理器來工作的。當事件源產生事件後,調用事件處理器相應的處理方式,一個事件就可以得到處理。在事件源調用事件處理器中特定方法的時候,還要傳遞給事件處理器相應事件的狀態信息,這樣事件處理器才能夠根據提供的事件信息來決定自己的行為。並且,它並不需要解析完整個文檔,在按內容順序解析文檔的過程中,SAX會判斷當前讀到的字元是否合法XML語法中的某部分,如果符合就會觸發事件。所謂事件,其實就是一些回調(callback)方法,這些方法(事件)定義在ContentHandler介面。

  在SAX介面中,事件源是org.xml.sax包中的XMLReader,它通過parser()方法來解析XML文檔,並產生事件。事件處理器是org.xml.sax包中ContentHandle、DTDHandler、ErrorHandler以及EntityResolver這4個介面。XMLReader通過相應事件處理器註冊方法setXXX()來完成的與ContentHandle、DTDHandler、ErrorHandler以及EntityResolver這4個介面的連接。

  什麼是事件驅動模式?它將XML文檔轉換成一系列的事件,由單獨的事件處理器來決定如何處理。一個可以產生事件的對象叫做事件源,而一個可以針對事件做出響應的對象就被叫做事件處理器。

  優點:不用實現調入整個文檔,占用資源少。尤其在嵌入式環境中,如android,極力推薦使用SAX解析。

  缺點:不像DOM解析一樣將文檔長期駐留在記憶體中,數據不是持久的。如果事件過後沒有保存數據,數據就會丟失。

  使用場合:機器有性能限制。

二、SAX解析步驟

         1.創建SAXParserFactory對象
SAXParserFactory spf = SAXParserFactory.newInstance();
         2.根據SAXParserFactory.newSAXParser()方法返回一個SAXParser解析器
SAXParser saxParser = spf.newSAXParser();
         3.實例為一個DefaultHandler對象
public class XMLContentHandler extends DefaultHandler {  

    //接收文檔開始的通知。當遇到文檔的開頭的時候,調用這個方法,可以在其中做一些預處理。   
    @Override   
     public void startDocument() throws SAXException {
        ...
    }
  //接收元素開始的通知。當讀到一個開始標簽的時候,會觸發這個方法。其中uri表示元素的命名空間;
  //localName表示元素的本地名稱(不帶首碼);qName表示元素的限定名(帶首碼);attrs表示元素的屬性集合。
    @Override
    public void startElement(String uri, String localName, String qName,
 Attributes attributes) throws SAXException {
        ...
    }
  //接收字元數據的通知。改方法用來處理在XML文件中讀到的內容,第一個參數用來存放文件的內容,後面兩個參數
  //是讀到的字元串在這個數組中的起始位置和長度。使用newSreing(ch,start,length)就可以獲取內容。
   @Override
    public void characters(char[] ch, int start, int length) throws SAXException {
        ...
    }
  //接收文檔的結尾的通知。在遇到結束標簽的時候,調用這個方法。其中,uri表示元素的命名空間;
  //localName表示元素的本地名稱(不帶首碼);name表示元素的限定名(帶首碼)。
  @Override
    public void endElement(String uri, String localName, String qName) throws SAXException {
        ...
    }
         4.調用SAXParser的parser方法從輸入源中獲取到XML數據
 saxParser.parse(inputStream, handler);
 inputStream.close();

        也可以使用XMLReader的parse方法從輸入源中獲取到XML數據。

  5.通過DefaultHandler返回我們需要的數據集合。
handler.getPersons();

三、SAX解析XML代碼

1.創建一個XML文件itcase.xml,並將其放在res/raw文件夾下。
<?xml version="1.0" encoding="UTF-8"?>
<persons>
    <person id="23">
        <name>liming</name>
        <age>30</age>
    </person>
    <person id="20">
        <name>lixiangmei</name>
        <age>25</age>
    </person>
</persons>

如果沒有raw文件夾,就在res文件夾下創建一個raw文件夾,並創建xml文件。

2.修改視圖
<Button
        android:id="@+id/sax_button1"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_centerHorizontal="true"
        android:layout_marginTop="@dimen/fab_margin"
        android:gravity="center_horizontal"
        android:text="@string/SAX" />
<Button
        android:id="@+id/sax_button2"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_centerHorizontal="true"
        android:layout_marginTop="@dimen/fab_margin"
        android:gravity="center_horizontal"
        android:text="@string/SAX" />
<TextView
        android:id="@+id/text"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="Hello World!" />
3.添加XMLContentHandler類
package com.zhangmiao.analyzexmldemo;

import org.xml.sax.Attributes;
import org.xml.sax.SAXException;
import org.xml.sax.helpers.DefaultHandler;

import java.util.ArrayList;
import java.util.List;

/**
 * Created by zhangmiao on 2016/12/13.
 */
public class XMLContentHandler extends DefaultHandler {

    private List<Person> persons = null;
    private Person currentPerson;
    private String tagName = null;

    public List<Person> getPersons() {
        return persons;
    }

    @Override
    public void startDocument() throws SAXException {
        persons = new ArrayList<>();
    }

    @Override
    public void startElement(String uri, String localName, String qName, Attributes attributes) throws SAXException {
        if (localName.equals("person")) {
            currentPerson = new Person();
            currentPerson.setId(Integer.parseInt(attributes.getValue("id")));
        }
        this.tagName = localName;
    }

    @Override
    public void characters(char[] ch, int start, int length) throws SAXException {
        if (tagName != null) {
            String data = new String(ch, start, length);
            if (tagName.equals("name")) {
                this.currentPerson.setName(data);
            } else if (tagName.equals("age")) {
                this.currentPerson.setAge(Short.parseShort(data));
            }
        }
    }

    @Override
    public void endElement(String uri, String localName, String qName) throws SAXException {
        if (localName.equals("person")) {
            persons.add(currentPerson);
            currentPerson = null;
        }
        this.tagName = null;
    }
}
4.添加AnalyzeSAM類
package com.zhangmiao.analyzexmldemo;

import org.xml.sax.InputSource;
import org.xml.sax.XMLReader;

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

import javax.xml.parsers.SAXParser;
import javax.xml.parsers.SAXParserFactory;

/**
 * Created by zhangmiao on 2016/12/14.
 */
public class AnalyzeSAX {

    public static List<Person> readXML(InputStream inputStream) {
        try {
            SAXParserFactory spf = SAXParserFactory.newInstance();
            SAXParser saxParser = spf.newSAXParser();
            XMLContentHandler handler = new XMLContentHandler();
            saxParser.parse(inputStream, handler);
            inputStream.close();
            return handler.getPersons();
        } catch (Exception e) {
            e.printStackTrace();
        }
        return null;
    }

    public static List<Person> readXML(InputSource inputSource) {
        try {
            SAXParserFactory spf = SAXParserFactory.newInstance();
            SAXParser saxParser = spf.newSAXParser();
            XMLReader reader = saxParser.getXMLReader();
            XMLContentHandler handler = new XMLContentHandler();
            reader.setContentHandler(handler);
            reader.parse(inputSource);
            inputSource.getByteStream().close();
            return handler.getPersons();
        } catch (Exception e) {
            e.printStackTrace();
        }
        return null;
    }
}
5.修改MainActivity類
package com.zhangmiao.analyzexmldemo;

import android.os.Bundle;
import android.support.design.widget.FloatingActionButton;
import android.support.design.widget.Snackbar;
import android.support.v7.app.AppCompatActivity;
import android.support.v7.widget.Toolbar;
import android.util.Log;
import android.view.View;
import android.view.Menu;
import android.view.MenuItem;
import android.widget.Button;
import android.widget.TextView;

import org.xml.sax.InputSource;

import java.io.BufferedWriter;
import java.io.File;
import java.io.FileOutputStream;
import java.io.InputStream;
import java.io.OutputStreamWriter;
import java.io.StringWriter;
import java.util.List;

import javax.xml.parsers.SAXParser;
import javax.xml.parsers.SAXParserFactory;

public class MainActivity extends AppCompatActivity implements View.OnClickListener {

    private static final String TAG = "AnalyzeXMLDemo";

    private TextView mTextView;

    private InputStream inputStream;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        Log.v(TAG, "onCreate");
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
        setSupportActionBar(toolbar);

        FloatingActionButton fab = (FloatingActionButton) findViewById(R.id.fab);
        fab.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                Snackbar.make(view, "Replace with your own action", Snackbar.LENGTH_LONG)
                        .setAction("Action", null).show();
            }
        });

        Button saxButton1 = (Button) findViewById(R.id.sax_button1);
        Button saxButton2 = (Button) findViewById(R.id.sax_button2);
        mTextView = (TextView) findViewById(R.id.text);

        saxButton1.setOnClickListener(this);
        saxButton2.setOnClickListener(this);
    }

    @Override
    public void onClick(View v) {
        String result = "";
        inputStream = getResources().openRawResource(R.raw.itcase);
        switch (v.getId()) {
            case R.id.sax_button1:
                result += "--------- SAX1 ---------" + "\n";
                if (inputStream == null) {
                    result = "inputStream is null";
                } else {
                    List<Person> personList = AnalyzeSAX.readXML(inputStream);
                    if (personList != null) {
                        for (int i = 0; i < personList.size(); i++) {
                            String message = "id = " + personList.get(i).getId() + " , name = " + personList.get(i).getName()
                                    + " , age = " + personList.get(i).getAge() + ".\n";
                            result += message;
                        }
                    }
                }
                mTextView.setText(result);
                break;
            case R.id.sax_button2:
                result += "--------- SAX2 ---------" + "\n";
                InputSource inputSource = new InputSource();
                inputSource.setByteStream(inputStream);
                if (inputSource == null) {
                    result = "inputStream is null";
                } else {
                    List<Person> personList = AnalyzeSAX.readXML(inputSource);
                    if (personList != null) {
                        for (int i = 0; i < personList.size(); i++) {
                            String message = "id = " + personList.get(i).getId() + " , name = " + personList.get(i).getName()
                                    + " , age = " + personList.get(i).getAge() + ".\n";
                            result += message;
                        }
                    }
                }
                mTextView.setText(result);
                break;
            default:
                break;
        }
    }

    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
        // Inflate the menu; this adds items to the action bar if it is present.
        getMenuInflater().inflate(R.menu.menu_main, menu);
        return true;
    }

    @Override
    public boolean onOptionsItemSelected(MenuItem item) {
        // Handle action bar item clicks here. The action bar will
        // automatically handle clicks on the Home/Up button, so long
        // as you specify a parent activity in AndroidManifest.xml.
        int id = item.getItemId();

        //noinspection SimplifiableIfStatement
        if (id == R.id.action_settings) {
            return true;
        }

        return super.onOptionsItemSelected(item);
    }
}

 參考文章:

http://www.open-open.com/lib/view/open1392780226397.html

http://www.cnblogs.com/weixing/archive/2013/08/07/3243366.html

http://www.tuicool.com/articles/IvQvyq


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 安卓指紋認證使用智能手機觸摸感測器對用戶進行身份驗證。Android Marshmallow(棉花糖)提供了一套API,使用戶很容易使用觸摸感測器。在Android Marshmallow之前訪問觸摸感測器的方法不是標準的。 本文地址:http://wuyudong.com/2016/12/15/3 ...
  • 一般接到任務後程式員們通常都開始著手進行技術調研了,但我這活是項目負責人。還有一大堆事情要先期準備,沒人能幫忙。 一、人力配置 考慮的之前已經有一波人搞了大半年,但沒有起色,先期也沒有太大人力需求,所以不指望這批人了。因為是外包收錢按人頭工時算,所以拉幾個自己兄弟填坑。 初期人力配置 服務端x 1、 ...
  • 一.Pull解析方法介紹 除了可以使用SAX和DOM解析XML文件,也可以使用Android內置的Pull解析器解析XML文件。Pull解析器的運行方式與SAX解析器相似。它也是事件觸發的。Pull解析方式讓應用程式完全控制文檔該怎麼樣被解析。比如開始和結束元素事件,使用parser.next()可 ...
  • 關於 swift 中的open ,public ,fileprivate,private, internal的區別 以下按照修飾關鍵字的訪問約束範圍 從約束的限定範圍大到小的排序進行說明 open,public,fileprivate,private,internal 這幾個修飾詞的作用是用於修飾訪 ...
  • 示例代碼 ...
  • 一.DOM解析方法介紹 DOM是基於樹形結構的節點或信息片段的集合,允許開發人員使用DOM API遍歷XML樹,檢索所需數據。分析該結構通常需要載入整個文檔和構造樹形結構,然後才可以檢索和更新節點信息。 Android完全支持DOM解析。利用DOM中的對象可以對XML文檔進行讀取、搜索、修改、添加和 ...
  • 由OpenDigg 出品的iOS開源項目周報第一期來啦。我們的iOS開源周報集合了OpenDigg一周來新收錄的優質的iOS開發方面的開源項目,方便iOS開發人員便捷的找到自己需要的項目工具等。 PYSearch 優雅的iOS搜索控制項 Sourcery 為Swift提供meta編程 CreditCa ...
  • 畫橫線/豎線 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...