W3Cschool
恭喜您成為首批注冊用戶
獲得88經(jīng)驗值獎勵
前面兩節(jié)我們對Android內置的Http請求方式:HttpURLConnection和HttpClient,本來以為OkHttp 已經(jīng)集成進來了,然后想講解下Okhttp的基本用法,后來發(fā)現(xiàn)還是要導第三方,算了,放到進階部分 吧,而本節(jié)我們來學習下Android為我們提供的三種解析XML數(shù)據(jù)的方案!他們分別是: SAX,DOM,PULL三種解析方式,下面我們就來對他們進行學習!
首先我們來看看XML數(shù)據(jù)的一些要求以及概念:
核心代碼:
SAX解析類:SaxHelper.java:
/**
* Created by Jay on 2015/9/8 0008.
*/
public class SaxHelper extends DefaultHandler {
private Person person;
private ArrayList<Person> persons;
//當前解析的元素標簽
private String tagName = null;
/**
* 當讀取到文檔開始標志是觸發(fā),通常在這里完成一些初始化操作
*/
@Override
public void startDocument() throws SAXException {
this.persons = new ArrayList<Person>();
Log.i("SAX", "讀取到文檔頭,開始解析xml");
}
/**
* 讀到一個開始標簽時調用,第二個參數(shù)為標簽名,最后一個參數(shù)為屬性數(shù)組
*/
@Override
public void startElement(String uri, String localName, String qName,
Attributes attributes) throws SAXException {
if (localName.equals("person")) {
person = new Person();
person.setId(Integer.parseInt(attributes.getValue("id")));
Log.i("SAX", "開始處理person元素~");
}
this.tagName = localName;
}
/**
* 讀到到內容,第一個參數(shù)為字符串內容,后面依次為起始位置與長度
*/
@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.person.setName(data);
Log.i("SAX", "處理name元素內容");
} else if (this.tagName.equals("age")) {
this.person.setAge(Integer.parseInt(data));
Log.i("SAX", "處理age元素內容");
}
}
}
/**
* 處理元素結束時觸發(fā),這里將對象添加到結合中
*/
@Override
public void endElement(String uri, String localName, String qName)
throws SAXException {
if (localName.equals("person")) {
this.persons.add(person);
person = null;
Log.i("SAX", "處理person元素結束~");
}
this.tagName = null;
}
/**
* 讀取到文檔結尾時觸發(fā),
*/
@Override
public void endDocument() throws SAXException {
super.endDocument();
Log.i("SAX", "讀取到文檔尾,xml解析結束");
}
//獲取persons集合
public ArrayList<Person> getPersons() {
return persons;
}
}
然后我們在MainActivity.java中寫上寫上這樣一個方法,然后要解析XML的時候調用下 就好了~
private ArrayList<Person> readxmlForSAX() throws Exception {
//獲取文件資源建立輸入流對象
InputStream is = getAssets().open("person1.xml");
//①創(chuàng)建XML解析處理器
SaxHelper ss = new SaxHelper();
//②得到SAX解析工廠
SAXParserFactory factory = SAXParserFactory.newInstance();
//③創(chuàng)建SAX解析器
SAXParser parser = factory.newSAXParser();
//④將xml解析處理器分配給解析器,對文檔進行解析,將事件發(fā)送給處理器
parser.parse(is, ss);
is.close();
return ss.getPersons();
}
一些其他的話:
嗯,對了,忘記給大家說下我們是定義下面這樣一個person1.xml文件,然后放到assets目錄下的! 文件內容如下:person1.xml
<?xml version="1.0" encoding="UTF-8"?>
<persons>
<person id = "11">
<name>SAX解析</name>
<age>18</age>
</person>
<person id = "13">
<name>XML1</name>
<age>43</age>
</person>
</persons>
我們是把三種解析方式都糅合到一個demo中,所以最后才貼全部的效果圖,這里的話,貼下打印的Log, 相信大家會對SAX解析XML流程更加明了:
另外,外面的空白文本也是文本節(jié)點哦!解析的時候也會走這些節(jié)點!
核心代碼:
DomHelper.java
/**
* Created by Jay on 2015/9/8 0008.
*/
public class DomHelper {
public static ArrayList<Person> queryXML(Context context)
{
ArrayList<Person> Persons = new ArrayList<Person>();
try {
//①獲得DOM解析器的工廠示例:
DocumentBuilderFactory dbFactory = DocumentBuilderFactory.newInstance();
//②從Dom工廠中獲得dom解析器
DocumentBuilder dbBuilder = dbFactory.newDocumentBuilder();
//③把要解析的xml文件讀入Dom解析器
Document doc = dbBuilder.parse(context.getAssets().open("person2.xml"));
System.out.println("處理該文檔的DomImplemention對象=" + doc.getImplementation());
//④得到文檔中名稱為person的元素的結點列表
NodeList nList = doc.getElementsByTagName("person");
//⑤遍歷該集合,顯示集合中的元素以及子元素的名字
for(int i = 0;i < nList.getLength();i++)
{
//先從Person元素開始解析
Element personElement = (Element) nList.item(i);
Person p = new Person();
p.setId(Integer.valueOf(personElement.getAttribute("id")));
//獲取person下的name和age的Note集合
NodeList childNoList = personElement.getChildNodes();
for(int j = 0;j < childNoList.getLength();j++)
{
Node childNode = childNoList.item(j);
//判斷子note類型是否為元素Note
if(childNode.getNodeType() == Node.ELEMENT_NODE)
{
Element childElement = (Element) childNode;
if("name".equals(childElement.getNodeName()))
p.setName(childElement.getFirstChild().getNodeValue());
else if("age".equals(childElement.getNodeName()))
p.setAge(Integer.valueOf(childElement.getFirstChild().getNodeValue()));
}
}
Persons.add(p);
}
} catch (Exception e) {e.printStackTrace();}
return Persons;
}
}
代碼分析:
從代碼我們就可以看出DOM解析XML的流程,先整個文件讀入Dom解析器,然后形成一棵樹, 然后我們可以遍歷節(jié)點列表獲取我們需要的數(shù)據(jù)!
使用PULL解析XML數(shù)據(jù)的流程:
核心代碼:
public static ArrayList<Person> getPersons(InputStream xml)throws Exception
{
//XmlPullParserFactory pullPaser = XmlPullParserFactory.newInstance();
ArrayList<Person> persons = null;
Person person = null;
// 創(chuàng)建一個xml解析的工廠
XmlPullParserFactory factory = XmlPullParserFactory.newInstance();
// 獲得xml解析類的引用
XmlPullParser parser = factory.newPullParser();
parser.setInput(xml, "UTF-8");
// 獲得事件的類型
int eventType = parser.getEventType();
while (eventType != XmlPullParser.END_DOCUMENT) {
switch (eventType) {
case XmlPullParser.START_DOCUMENT:
persons = new ArrayList<Person>();
break;
case XmlPullParser.START_TAG:
if ("person".equals(parser.getName())) {
person = new Person();
// 取出屬性值
int id = Integer.parseInt(parser.getAttributeValue(0));
person.setId(id);
} else if ("name".equals(parser.getName())) {
String name = parser.nextText();// 獲取該節(jié)點的內容
person.setName(name);
} else if ("age".equals(parser.getName())) {
int age = Integer.parseInt(parser.nextText());
person.setAge(age);
}
break;
case XmlPullParser.END_TAG:
if ("person".equals(parser.getName())) {
persons.add(person);
person = null;
}
break;
}
eventType = parser.next();
}
return persons;
}
使用Pull生成xml數(shù)據(jù)的流程:
核心代碼:
public static void save(List<Person> persons, OutputStream out) throws Exception {
XmlSerializer serializer = Xml.newSerializer();
serializer.setOutput(out, "UTF-8");
serializer.startDocument("UTF-8", true);
serializer.startTag(null, "persons");
for (Person p : persons) {
serializer.startTag(null, "person");
serializer.attribute(null, "id", p.getId() + "");
serializer.startTag(null, "name");
serializer.text(p.getName());
serializer.endTag(null, "name");
serializer.startTag(null, "age");
serializer.text(p.getAge() + "");
serializer.endTag(null, "age");
serializer.endTag(null, "person");
}
serializer.endTag(null, "persons");
serializer.endDocument();
out.flush();
out.close();
}
運行效果圖:
代碼下載: XMLParseDemo.zip:下載 XMLParseDemo.zip
本節(jié)介紹了Android中三種常用的XML解析方式,DOM,SAX和PULL,移動端我們建議用后面這 兩種,而PULL用起來更加簡單,這里就不多說了,代碼是最好的老師~本節(jié)就到這里,下節(jié)我們 來學習Android為我們提供的扣腳Json解析方式!謝謝~
Copyright©2021 w3cschool編程獅|閩ICP備15016281號-3|閩公網(wǎng)安備35020302033924號
違法和不良信息舉報電話:173-0602-2364|舉報郵箱:jubao@eeedong.com
掃描二維碼
下載編程獅App
編程獅公眾號
聯(lián)系方式:
更多建議: