android操作XML的幾種方法總結(jié)
XML作為一種業(yè)界公認(rèn)的數(shù)據(jù)交換格式,在各個(gè)平臺與語言之上,都有廣泛使用和實(shí)現(xiàn)。其標(biāo)準(zhǔn)型,可靠性,安全性......毋庸置疑。在android平臺上,我們要想實(shí)現(xiàn)數(shù)據(jù)存儲(chǔ)和數(shù)據(jù)交換,經(jīng)常會(huì)使用到xml數(shù)據(jù)格式和xml文件。
小提示:android中存儲(chǔ)數(shù)據(jù)一般有如下幾種:SharedPreferences(參數(shù)化),XML文件,sqllite數(shù)據(jù)庫,網(wǎng)絡(luò),ContentProvider(內(nèi)容提供者)等。
在android中,操作xml文件,一般有幾種方式:SAX操作,Pull操作,DOM操作等。其中DOM的方式,可能是大家最熟悉的,也是符合W3C標(biāo)準(zhǔn)的。
1)
在java平臺中,有諸如DOM4J這樣優(yōu)秀的開源包,極大程度的方便大家使用DOM標(biāo)準(zhǔn)來操作XML文件。在javascript中,不同的瀏覽器解析引擎,對DOM的解析和操作也略有差異(不過這不是本章介紹的重點(diǎn))。而DOM的方式,也有其缺點(diǎn)。通常一次性加載xml文件,再使用DOM的 api去進(jìn)行解析,這樣很大程度的消耗內(nèi)存,對性能會(huì)有一定影響。而我們的android手機(jī),雖然配置在不斷的升級,但是內(nèi)存方面,暫時(shí)還無法與傳統(tǒng)的PC去媲美。所以,在android上面,不太推薦使用DOM的方式來解析和操作XML。
package cn.itcast.service;
import java.io.InputStream;
import java.util.ArrayList;
import java.util.List;
import javax.xml.parsers.DocumentBuilder;
import javax.xml.parsers.DocumentBuilderFactory;
import org.w3c.dom.Document;
import org.w3c.dom.Element;
import org.w3c.dom.Node;
import org.w3c.dom.NodeList;
import cn.itcast.model.Person;
public class DomPersonService {
public List<Person> getPersons(InputStream stream) throws Throwable
{
List<Person> list =new ArrayList<Person>();
DocumentBuilderFactory factory =DocumentBuilderFactory.newInstance();
DocumentBuilder builder =factory.newDocumentBuilder();
Document dom = builder.parse(stream);//解析完成,并以dom樹的方式存放在內(nèi)存中。比較消耗性能
//開始使用dom的api去解析
Element root = dom.getDocumentElement();//根元素
NodeList personNodes = root.getElementsByTagName("person");//返回所有的person元素節(jié)點(diǎn)
//開始遍歷啦
for(int i=0;i<personNodes.getLength();i++)
{
Person person =new Person();
Element personElement =(Element)personNodes.item(i);
person.setId(new Integer( personElement.getAttribute("id")));//將person元素節(jié)點(diǎn)的屬性節(jié)點(diǎn)id的值,賦給person對象
NodeList personChildrenNodes =personElement.getChildNodes();//獲取person節(jié)點(diǎn)的所有子節(jié)點(diǎn)
//遍歷所有子節(jié)點(diǎn)
for(int j=0;j<personChildrenNodes.getLength();j++)
{
//判斷子節(jié)點(diǎn)是否是元素節(jié)點(diǎn)(如果是文本節(jié)點(diǎn),可能是空白文本,不處理)
if(personChildrenNodes.item(j).getNodeType()==Node.ELEMENT_NODE)
{
//子節(jié)點(diǎn)--元素節(jié)點(diǎn)
Element childNode =(Element)personChildrenNodes.item(j);
if("name".equals(childNode.getNodeName()))
{
//如果子節(jié)點(diǎn)的名稱是“name”.將子元素節(jié)點(diǎn)的第一個(gè)子節(jié)點(diǎn)的值賦給person對象
person.setName(childNode.getFirstChild().getNodeValue());
}else if("age".equals(childNode.getNodeValue()))
{
person.setAge(new Integer(childNode.getFirstChild().getNodeValue()));
}
}
}
list.add(person);
}
return list;
}
}
2)
SAX(Simple API for XML),是一個(gè)使用非常廣泛的XML解析標(biāo)準(zhǔn),通常使用Handler模式來處理XML文檔,這種處理模式和我們平常習(xí)慣的理解方式很不同,身邊也經(jīng)常有一些朋友在剛接觸SAX的時(shí)候會(huì)覺得理解起來有些困難。其實(shí)SAX并不復(fù)雜,只不過是換了一種思維方式,正如它的名字所表示的,為了讓我們以更簡單的方式來處理XML文檔,下面我們就開始吧。
package cn.itcast.service;
import java.io.InputStream;
import java.util.ArrayList;
import java.util.List;
import javax.xml.parsers.SAXParser;
import javax.xml.parsers.SAXParserFactory;
import org.xml.sax.Attributes;
import org.xml.sax.SAXException;
import org.xml.sax.helpers.DefaultHandler;
import cn.itcast.model.Person;
public class SAXPersonService {
public List<Person> getPersons(InputStream inStream) throws Throwable
{
SAXParserFactory factory = SAXParserFactory.newInstance();//工廠模式還是單例模式?
SAXParser parser =factory.newSAXParser();
PersonParse personParser =new PersonParse();
parser.parse(inStream, personParser);
inStream.close();
return personParser.getPerson();
}
private final class PersonParse extends DefaultHandler
{
private List<Person> list = null;
Person person =null;
private String tag=null;
public List<Person> getPerson() {
return list;
}
@Override
public void startDocument() throws SAXException {
list =new ArrayList<Person>();
}
@Override
public void startElement(String uri, String localName, String qName,
Attributes attributes) throws SAXException {
if("person".equals(localName))
{
//xml元素節(jié)點(diǎn)開始時(shí)觸發(fā),是“person”
person = new Person();
person.setId(new Integer(attributes.getValue(0)));
}
tag =localName;//保存元素節(jié)點(diǎn)名稱
}
@Override
public void endElement(String uri, String localName, String qName)
throws SAXException {
//元素節(jié)點(diǎn)結(jié)束時(shí)觸發(fā),是“person”
if("person".equals(localName))
{
list.add(person);
person=null;
}
tag =null;//結(jié)束時(shí),需要清空tag
}
@Override
public void characters(char[] ch, int start, int length)
throws SAXException {
if(tag!=null)
{
String data = new String(ch,start,length);
if("name".equals(tag))
{
person.setName(data);
}else if("age".equals(tag))
{
person.setAge(new Integer(data));
}
}
}
}
}
3)
Pull解析和Sax解析很相似,都是輕量級的解析,在Android的內(nèi)核中已經(jīng)嵌入了Pull,所以我們不需要再添加第三方j(luò)ar包來支持Pull。Pull解析和Sax解析不一樣的地方有(1)pull讀取xml文件后觸發(fā)相應(yīng)的事件調(diào)用方法返回的是數(shù)字(2)pull可以在程序中控制想解析到哪里就可以停止解析。
package cn.itcast.service;
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.XmlSerializer;
import android.util.Xml;
import cn.itcast.model.Person;
public class PullPersonService {
//保存xml文件
public static void saveXML(List<Person> list,Writer write)throws Throwable
{
XmlSerializer serializer =Xml.newSerializer();//序列化
serializer.setOutput(write);//輸出流
serializer.startDocument("UTF-8", true);//開始文檔
serializer.startTag(null, "persons");
//循環(huán)去添加person
for (Person person : list) {
serializer.startTag(null, "person");
serializer.attribute(null, "id", person.getId().toString());//設(shè)置id屬性及屬性值
serializer.startTag(null, "name");
serializer.text(person.getName());//文本節(jié)點(diǎn)的文本值--name
serializer.endTag(null, "name");
serializer.startTag(null, "age");
serializer.text(person.getAge().toString());//文本節(jié)點(diǎn)的文本值--age
serializer.endTag(null, "age");
serializer.endTag(null, "person");
}
serializer.endTag(null, "persons");
serializer.endDocument();
write.flush();
write.close();
}
public List<Person> getPersons(InputStream stream) throws Throwable
{
List<Person> list =null;
Person person =null;
XmlPullParser parser =Xml.newPullParser();
parser.setInput(stream,"UTF-8");
int type =parser.getEventType();//產(chǎn)生第一個(gè)事件
//只要當(dāng)前事件類型不是”結(jié)束文檔“,就去循環(huán)
while(type!=XmlPullParser.END_DOCUMENT)
{
switch (type) {
case XmlPullParser.START_DOCUMENT:
list = new ArrayList<Person>();
break;
case XmlPullParser.START_TAG:
String name=parser.getName();//獲取解析器當(dāng)前指向的元素名稱
if("person".equals(name))
{
person =new Person();
person.setId(new Integer(parser.getAttributeValue(0)));
}
if(person!=null)
{
if("name".equals(name))
{
person.setName(parser.nextText());//獲取解析器當(dāng)前指向的元素的下一個(gè)文本節(jié)點(diǎn)的文本值
}
if("age".equals(name))
{
person.setAge(new Integer(parser.nextText()));
}
}
break;
case XmlPullParser.END_TAG:
if("person".equals(parser.getName()))
{
list.add(person);
person=null;
}
break;
}
type=parser.next();//這句千萬別忘了哦
}
return list;
}
}
下面是Model層的Person類的代碼:
package cn.itcast.model;
public class Person {
private Integer id;
public Integer getId() {
return id;
}
public void setId(Integer id) {
this.id = id;
}
private String name;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
private Integer age;
public Integer getAge() {
return age;
}
public void setAge(Integer age) {
this.age = age;
}
public Person()
{
}
public Person(Integer id, String name, Integer age) {
this.id = id;
this.name = name;
this.age = age;
}
@Override
public String toString() {
return "Person [id=" + id + ", name=" + name + ", age=" + age + "]";
}
}
相關(guān)文章
Android短信驗(yàn)證碼倒計(jì)時(shí)驗(yàn)證的2種常用方式
各位開發(fā)者們在開發(fā)中經(jīng)常會(huì)遇到獲取短信驗(yàn)證碼,獲取驗(yàn)證碼后需要等待1分鐘倒計(jì)時(shí),這段時(shí)間是不能再次發(fā)送短信請求的。這篇文章總結(jié)了兩種常用的Android​短信驗(yàn)證碼倒計(jì)時(shí)驗(yàn)證方式,有需要的朋友們可以參考借鑒,下面來一起看看吧。2016-12-12
深入分析Android ViewStub的應(yīng)用詳解
本篇文章是對Android ViewStub的應(yīng)用進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-05-05
Android在Sqlite3中的應(yīng)用及多線程使用數(shù)據(jù)庫的建議(實(shí)例代碼)
這篇文章主要介紹了Android在Sqlite3中的應(yīng)用及多線程使用數(shù)據(jù)庫的建議,包括編寫數(shù)據(jù)庫具體操作類、增刪改查,通過實(shí)例代碼介紹了在實(shí)際中的應(yīng)用,需要的朋友可以參考下2022-04-04
ComposeDesktop開發(fā)桌面端多功能APK工具
這篇文章主要為大家介紹了ComposeDesktop開發(fā)桌面端多功能APK工具實(shí)現(xiàn)示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-07-07
android LabelView實(shí)現(xiàn)標(biāo)簽云效果
這篇文章主要為大家詳細(xì)介紹了android LabelView實(shí)現(xiàn)標(biāo)簽云效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-05-05
Android實(shí)現(xiàn)平滑翻動(dòng)效果
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)平滑翻動(dòng)效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-04-04
基于Flutter實(shí)現(xiàn)短信驗(yàn)證碼監(jiān)控與轉(zhuǎn)發(fā)
這篇文章主要為大家詳細(xì)介紹了如何基于Flutter實(shí)現(xiàn)短信驗(yàn)證碼監(jiān)控與轉(zhuǎn)發(fā)功能,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2025-03-03
Android編程實(shí)現(xiàn)手機(jī)震動(dòng)功能的方法
這篇文章主要介紹了Android編程實(shí)現(xiàn)手機(jī)震動(dòng)功能的方法,結(jié)合實(shí)例形式分析了Android實(shí)現(xiàn)手機(jī)震動(dòng)功能的核心代碼與權(quán)限控制操作技巧,需要的朋友可以參考下2017-06-06
Android使用Xutils3進(jìn)行斷點(diǎn)下載的實(shí)例
在本篇內(nèi)容中小編給各位整理了關(guān)于Android使用Xutils3進(jìn)行斷點(diǎn)下載的實(shí)例以及相關(guān)代碼,需要的朋友們參考下。2019-07-07

