Android之SAX解析XML檔案,androidsaxxml
一.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