本篇博客重点介绍Android中三种解析XML的方式,包括PULL、SAX、DOM,当然不止这些,还可以用第三方的jar包提供的解析,只是这三种在Android中比较常用吧。再顺便介绍一下AndroidTestCase的用法,用来测试所写的解析业务逻辑是否正确。

本篇博客使用的xml文件如下:

student.xml

<?xml version="1.0" encoding="UTF-8"?>
<students>  <student id="1003">  <name>ZhangSan</name>  <age>23</age><score>89</score>  </student><student id="1004">  <name>LiSi</name>  <age>24</age><score>72</score>  </student><student id="1005">  <name>WangWu</name>  <age>25</age><score>79</score>  </student>
</students>

各字体属性应该很清楚了,这里不再介绍,也不是重点。

此xml文件放在Android工程下面的assets目录下面,等待解析。。。

再建一个类Student.java

package com.and.xml;public class Student {private int id;private String name;private int age;private float score;public Student() {super();}public Student(int id, String name, int age, float score) {super();this.id = id;this.name = name;this.age = age;this.score = score;}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 int getAge() {return age;}public void setAge(int age) {this.age = age;}public float getScore() {return score;}public void setScore(float score) {this.score = score;}@Overridepublic String toString() {// TODO Auto-generated method stubreturn "Id:" + this.id + ",Name:" + this.name + ",Age" + this.age+ ",Score:" + this.score;}
}

下面分别介绍三种解析方式。

第一种:PULL解析

PullParseService

package com.and.xml;import java.io.InputStream;
import java.util.ArrayList;
import java.util.List;import org.xmlpull.v1.XmlPullParser;
import org.xmlpull.v1.XmlPullParserFactory;
/*** PULL解析示例* @author Administrator**/
public class PullParseService {public static List<Student> getStudents(InputStream input) throws Exception {List<Student> data = null;Student stu = null;XmlPullParserFactory fac = XmlPullParserFactory.newInstance();fac.setNamespaceAware(true);XmlPullParser parser = fac.newPullParser();parser.setInput(input, "UTF-8");int eventType = parser.getEventType();while (eventType != XmlPullParser.END_DOCUMENT) {switch (eventType) {case XmlPullParser.START_DOCUMENT:System.out.println("START_DOCUMENT");data = new ArrayList<Student>();break;case XmlPullParser.START_TAG:if ("student".equals(parser.getName())) {stu = new Student();stu.setId(Integer.parseInt(parser.getAttributeValue(0)));}if (stu != null) {if ("name".equals(parser.getName())) {stu.setName(parser.nextText());} else if ("age".equals(parser.getName())) {stu.setAge(Integer.parseInt(parser.nextText()));} else if ("score".equals(parser.getName())) {stu.setScore(Float.parseFloat(parser.nextText()));}}break;case XmlPullParser.END_TAG:if ("student".equals(parser.getName())) {if (data != null && stu != null) {data.add(stu);stu = null;}}break;}eventType = parser.next();// 注意:此处勿要写成parser.next();不要理解成指针}return data;}
}

至此,PULL解析的核心业务完成了,怎样来测试有没有问题呢?一般情况下,都是在Activity输出调试日志,根据调试日志判断是否解析成功。这里换一种方式,用Android的测试用例来测试一下。

TestParseService.java

package com.and.test;import java.io.InputStream;
import java.util.List;import javax.xml.parsers.SAXParserFactory;import org.xml.sax.InputSource;
import org.xml.sax.XMLReader;import com.and.xml.DomParseService;
import com.and.xml.PullParseService;
import com.and.xml.SaxParserService;
import com.and.xml.Student;import android.test.AndroidTestCase;
import android.util.Log;/*** 测试三种解析方式(Pull、SAX、Dom)* * @author And 2012-02-29*/
public class TestParseService extends AndroidTestCase {private static final String TAG = "testService";InputStream input;List<Student> students;public void init() throws Exception {input = this.getContext().getAssets().open("students.xml");}// 测试Pull解析方式public void testPull() throws Exception {init();students = PullParseService.getStudents(input);for (Student stu : students) {Log.i(TAG, stu.toString());}}// 测试SAX解析方式public void testSAX() throws Exception {init();SAXParserFactory fac = SAXParserFactory.newInstance();XMLReader reader = fac.newSAXParser().getXMLReader();SaxParserService saxHandler = new SaxParserService();reader.setContentHandler(saxHandler);reader.parse(new InputSource(input));students = saxHandler.getParseData();for (Student stu : students) {Log.i(TAG, stu.toString());}}// 测试DOM解析方式public void testDom() throws Exception {init();students = DomParseService.getPersonsByParseXml(input);for (Student stu : students) {Log.i(TAG, stu.toString());}}}

注意一定要继承自 AndroidTestCase这个类。这个文件中写了所有三种解析的测试方法,其它的忽视吧,只看testPull方法,它是用来测试上面所写的PULL解析业务的 。

那么怎样测试呢?

鼠标选中testPull方法名——>右键——>Run As——>Anroid JUnit Test

会提示以下错误:

大概意思就是没有配置running tests.控制台输出:

XmlParseDemo does not specify a android.test.InstrumentationTestRunner instrumentation or does not declare uses-library android.test.runner in its AndroidManifest.xml

从上面的提示信息可知,需要在AndroidManifest.xml中作一些配置,包括instrumentation和uses-library的配置

在AndroidManifest.xml文件中添加如下两行

<instrumentation android:name="android.test.InstrumentationTestRunner" android:targetPackage="com.and.pull"></instrumentation>
<uses-library android:name="android.test.runner"/>

注意添加位置:

第一句跟application节点同级。

第二句跟activity同级。

上面介绍的方法是手动代码添加,下面介绍一种图形化的方式,只需要点击鼠标就可以搞定。

打开AndroidManifest.xml文件

点击Add...

这样,use-library就添加好了

同样的方法添加instrumentation属性

注意Target package后面的内容:com.and.xml包

整个工程目录结构如图:

然后查看一下AndroidManifest.xml的内容,已经包含了刚才添加的那两句了:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"package="com.and.xml"android:versionCode="1"android:versionName="1.0" ><uses-sdk android:minSdkVersion="7" /><instrumentation android:targetPackage="com.and.test" android:name="android.test.InstrumentationTestRunner"></instrumentation><applicationandroid:icon="@drawable/ic_launcher"android:label="@string/app_name" ><activityandroid:label="@string/app_name"android:name="com.and.xml.MainActivity" ><intent-filter ><action android:name="android.intent.action.MAIN" /><category android:name="android.intent.category.LAUNCHER" /></intent-filter></activity><uses-library android:name="android.test.runner"/></application></manifest>

OK,然后继续之前的操作“鼠标选中testPull方法名——>右键——>Run As——>Anroid JUnit Test”

如果出现类似这样的页面,就表示测试用例建立成功,并且测试方法通过


左上角的绿色条,表示测试方法通过,右下角的调试日志输出,通过判断可以知道解析成功。

第二种:SAX解析

SaxParserService.java

package com.and.xml;import java.util.ArrayList;
import java.util.List;import org.xml.sax.Attributes;
import org.xml.sax.SAXException;
import org.xml.sax.helpers.DefaultHandler;/*** SAX解析示例* * @author Administrator* */
public class SaxParserService extends DefaultHandler {List<Student> data = null;Student stu = null;String tag = "";@Overridepublic void characters(char[] ch, int start, int length) throws SAXException {super.characters(ch, start, length);if (stu != null) {String str = new String(ch, start, length);if (tag.equals("name")) {stu.setName(str);} else if (tag.equals("age")) {stu.setAge(Integer.parseInt(str));} else if (tag.equals("score")) {stu.setScore(Float.parseFloat(str));}}}@Overridepublic void endDocument() throws SAXException {super.endDocument();}@Overridepublic void endElement(String uri, String localName, String qName)throws SAXException {super.endElement(uri, localName, qName);if (localName.equals("student") && stu != null) {data.add(stu);stu = null;}tag = "";}@Overridepublic void startDocument() throws SAXException {super.startDocument();data = new ArrayList<Student>();}@Overridepublic void startElement(String uri, String localName, String qName,Attributes attributes) throws SAXException {super.startElement(uri, localName, qName, attributes);tag = localName;if (localName.equals("student")) {stu = new Student();}if (attributes.getValue(0) != null) {stu.setId(Integer.parseInt(attributes.getValue(0)));}}public List<Student> getParseData() {return data;}}

注意一定要继承自DefaultHandler,然复写里面的方法,这些方法名字根据字面意思很容易理解它的作用。

然后通过上面的的测试文件,按照类似的方法测试一下testSAX()方法,如果出现绿条和日志输出的话,表明解析业务逻辑成功。

第三种:DOM解析

DomParseService.java

package com.and.xml;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;/*** DOM解析示例*/
/*** DOM解析器在解析XML文档时,会把文档中的所有元素,按照其出现的层次关系,解析成一个个Node对象(节点)。Node对象提供了一系列常量来代表结点的类型* ,当开发人员获得某个Node类型后,就可以把Node节点转换成相应节点对象(Node的子类对象),以便于调用其特有的方法。* Node对象提供了相应的方法去获得它的父结点或子结点。编程人员通过这些方法就可以读取整个XML文档的内容、或添加、修改、删除XML文档的内容.* * 缺点: 一次性的完全加载整个xml文件,需要消耗大量的内存。*/
public class DomParseService {public static List<Student> getPersonsByParseXml(InputStream is) throws Exception {List<Student> persons = new ArrayList<Student>();DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();try {DocumentBuilder builder = factory.newDocumentBuilder();Document document = builder.parse(is);Element root = document.getDocumentElement();NodeList items = root.getElementsByTagName("student");// 得到所有person节点for (int i = 0; i < items.getLength(); i++) {Student Student = new Student();Element personNode = (Element) items.item(i);Student.setId(new Integer(personNode.getAttribute("id")));// 获取person节点下的所有子节点(标签之间的空白节点和name/age元素)NodeList childsNodes = personNode.getChildNodes();for (int j = 0; j < childsNodes.getLength(); j++) {Node node = (Node) childsNodes.item(j); // 判断是否为元素类型if (node.getNodeType() == Node.ELEMENT_NODE) {Element childNode = (Element) node;// 判断是否name元素if ("name".equals(childNode.getNodeName())) {// 获取name元素下Text节点,然后从Text节点获取数据Student.setName(childNode.getFirstChild().getNodeValue());} else if ("age".equals(childNode.getNodeName())) {Student.setAge(new Short(childNode.getFirstChild().getNodeValue()));} else if ("score".equals(childNode.getNodeName())) {Student.setScore(Float.parseFloat(childNode.getFirstChild().getNodeValue()));}}}persons.add(Student);}is.close();} catch (Exception e) {e.printStackTrace();}return persons;}
}

类似的测试方法。。。

至此,三种解析方式全部完成了,如果分别测试这三种方法的时候,一路绿条的话,那么恭喜,解析业务逻辑成功。否则,可能还有哪里有问题,请仔细检查。

对比这三种解析方式,我个人认为PULL和SAX解析方式类似,都是事件触发型的,就是当解析到某个节点的时候触发相应的事件。说明一下DOM解析,会把文档中的所有元素,按照其出现的层次关系,解析成一个个Node对象(节点),可见它会有点占内存,但是如果待解析的xml文件相对较小的话,使用DOM解析 优点还是很明确的。

Android解析XML三种方式(PULL、SAX、DOM)相关推荐

  1. 解析XML三种方式(PULL、SAX、DOM)

    本篇博客重点介绍Android中三种解析XML的方式,包括PULL.SAX.DOM,当然不止这些,还可以用第三方的jar包提供的解析,只是这三种在Android中比较常用吧.再顺便介绍一下Androi ...

  2. adroid解析XML三种方式

    废话少说,直接上代码: Pull 解析     跳至 [1] [2] [3] [全屏预览] ? 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 2 ...

  3. android xml解析的三种方式

    2019独角兽企业重金招聘Python工程师标准>>> 在android开发中,经常用到去解析xml文件,常见的解析xml的方式有一下三种:SAX.Pull.Dom解析方式.最近做了 ...

  4. XML解析的三种方式(dom,sax,dom4j)

    1.Dom解析: 要解析的xml文件内容: <?xml version="1.0" encoding="utf-8" standalone="n ...

  5. xml文件解析的三种方式

    SAX解析XML SAX是一个解析速度快并且占用内存少的xml解析器,非常适合用于Android等移动设备. SAX解析XML文件采用的是事件驱动,也就是说,它并不需要解析完整个文档,在按内容顺序解析 ...

  6. 爬虫之数据解析的三种方式

    一,正则表达式解析 re正则就不写了,前面已经写入一篇很详细的正则表达式模块了~ 而且,在爬虫中,下面两种方式用的多一些~ 正则表达式:https://www.cnblogs.com/peng104/ ...

  7. Python数据解析的三种方式

    一,正则表达式解析 在爬虫中,下面两种方式用的多一些~ 大致用法: pattern = re.compile('<dd>.*?board-index.*?>(\d+)</i&g ...

  8. 爬虫数据解析的三种方式

    一.正则表达式解析 常用正则表达式回顾:单字符:. : 除换行以外所有字符[] :[aoe] [a-w] 匹配集合中任意一个字符\d :数字 [0-9]\D : 非数字\w :数字.字母.下划线.中文 ...

  9. 爬虫之网络数据解析的三种方式---正则表达式、XPath数据解析库、BeautifulSoup数据解析库

    1.正则表达式 爬虫的四个主要步骤: 明确目标(要知道你准备取哪个范围或者网站上取搜索) 爬(将所有网站的内容全部爬下来) 取(取掉对我们没用的数据) 处理数据(按照我们想要的方式存储和使用) 什么是 ...

最新文章

  1. PAT甲级1032 Sharing :[C++题解]链表
  2. 计算机技术应用在教学中的优势,[浅谈多媒体在教学中的应用及优势] 多媒体教学的优势...
  3. java环境的意义_java环境变量配置的意义
  4. 《犯罪心理学》读书笔记(part8)--犯罪心理的主观差异(中下)
  5. 一些培养程序员leadership的经验教训
  6. Apache Camel简介
  7. Flask 模板 之 变量和过滤器
  8. 又一个学期结束了,送给在校大学生几句话
  9. PHP消息队列实现及应用
  10. local host 和 IP 相关的配置文件。
  11. HTML5期末大作业:动漫A网站设计——动画漫展学习资料电影模板(6页) 网页设计作业 / 动漫网页设计作业,网页设计作业 / 动漫网页设计成品,网页设计作业 / 动漫网页设计成品模板下载
  12. MTD系列 - 关于linux MTD的一些理解
  13. C#生成条形码图片的简单方法
  14. 最高效的七个云原生开发原则
  15. 【Ubuntu 16.04中为Chromium、Chrome、Firefox安装Flash播放器插件教程】
  16. 《途客圈创业记:不疯魔,不成活》一一2.4 与iWeekend再续前缘
  17. Python - 3.6 学习二
  18. webstorm实用技巧
  19. 用户绑定QQ邮箱找回密码
  20. 企业薪酬结构的选择方法

热门文章

  1. excel取消密码_如何将Excel设定为只能填写不能修改的模式?
  2. 目前手机号码运营商正则判断功能
  3. 香农公式说明了什么_香农公式意义
  4. 拿捏SQL数据分析:从基础破冰到面试题解
  5. Win10启动windows audio服务失败解决方案
  6. 如何建立一个网站,可用互联网访问?(原创详细教程)
  7. [The Preliminary Contest for ICPC Asia Xuzhou 2019 - 徐州网络赛E] XKC's basketball team
  8. 项目实施指南(仅供学习使用)
  9. 生活娱乐 岛城将开首家机器人餐厅
  10. 剑指offer:Python 把字符串转换成整数