本篇博客重点介绍Android中三种解析XML的方式,包括PULL、SAX、DOM,当然不止这些,还可以用第三方的jar包提供的解析,只是这三种在Android中比较常用吧。再顺便介绍一下AndroidTestCase的用法,用来测试所写的解析业务逻辑是否正确。
本篇博客使用的xml文件如下:
student.xml
- <?xmlversion="1.0"encoding="UTF-8"?>
- <students>
- studentid="1003">
- name>ZhangSan</age>23score>89studentstudentid="1004">LiSi>24>72studentid="1005">WangWu>25>79>
各字体属性应该很清楚了,这里不再介绍,也不是重点。
此xml文件放在Android工程下面的assets目录下面,等待解析。。。
再建一个类Student.java
packagecom.and.xml;
-
- publicclassStudent{
- privateintid;
- privateStringname;
- intage;
- floatscore;
-
- publicStudent(){
- super();
- }
- publicStudent(intid,Stringname,intage,153); font-weight:bold; background-color:inherit">floatscore){
- this.id=id;
- this.name=name;
- this.age=age;
- this.score=score;
- intgetId(){
- returnid;
- voidsetId(intid){
- this.id=id;
- publicStringgetName(){
- returnname;
- voidsetName(Stringname){
- intgetAge(){
- returnage;
- voidsetAge(intage){
- this.age=age;
- floatgetScore(){
- returnscore;
- voidsetScore( @Override
- publicStringtoString(){
-
- return"Id:"+this.id+",Name:"+this.name+",Age"+this.age
- +",Score:"+this.score;
- }
- }
下面分别介绍三种解析方式。
第一种:PULL解析
PullParseService
importjava.io.InputStream;
- importjava.util.ArrayList;
- importjava.util.List;
- importorg.xmlpull.v1.XmlPullParser;
- importorg.xmlpull.v1.XmlPullParserFactory;
-
- *PULL解析示例
- *@authorAdministrator
- *
- */
- classPullParseService{
- staticList<Student>getStudents(InputStreaminput)throwsException{
- List<Student>data=null;
- Studentstu=null;
- XmlPullParserFactoryfac=XmlPullParserFactory.newInstance();
- fac.setNamespaceAware(true);
- XmlPullParserparser=fac.newPullParser();
- parser.setInput(input,"UTF-8");
- inteventType=parser.getEventType();
- while(eventType!=XmlPullParser.END_DOCUMENT){
- switch(eventType){
- caseXmlPullParser.START_DOCUMENT:
- System.out.println("START_DOCUMENT");
- data=newArrayList<Student>();
- break;
- caseXmlPullParser.START_TAG:
- if("student".equals(parser.getName())){
- stu=newStudent();
- stu.setId(Integer.parseInt(parser.getAttributeValue(0)));
- if(stu!=null){
- if("name".equals(parser.getName())){
- stu.setName(parser.nextText());
- }elseif("age".equals(parser.getName())){
- stu.setAge(Integer.parseInt(parser.nextText()));
- if("score".equals(parser.getName())){
- stu.setScore(Float.parseFloat(parser.nextText()));
- break;
- caseXmlPullParser.END_TAG:
- if("student".equals(parser.getName())){
- if(data!=null&&stu!= data.add(stu);
- stu= eventType=parser.next();
- returndata;
- }
至此,PULL解析的核心业务完成了,怎样来测试有没有问题呢?一般情况下,都是在Activity输出调试日志,根据调试日志判断是否解析成功。这里换一种方式,用Android的测试用例来测试一下。
TestParseService.java
copy
packagecom.and.test;
- importjava.util.List;
- importjavax.xml.parsers.SAXParserFactory;
- importorg.xml.sax.InputSource;
- importorg.xml.sax.XMLReader;
- importcom.and.xml.DomParseService;
- importcom.and.xml.PullParseService;
- importcom.and.xml.SaxParserService;
- importcom.and.xml.Student;
- importandroid.test.AndroidTestCase;
- importandroid.util.Log;
- *测试三种解析方式(Pull、SAX、Dom)
- *
- *@authorAnd2012-02-29
- classTestParseServiceextendsAndroidTestCase{
- staticfinalStringTAG="testService";
- InputStreaminput;
- List<Student>students;
- voidinit() input=this.getContext().getAssets().open("students.xml");
- //测试Pull解析方式
- voidtestPull()throwsException{
- init();
- students=PullParseService.getStudents(input);
- for(Studentstu:students){
- Log.i(TAG,stu.toString());
- //测试SAX解析方式
- voidtestSAX() init();
- SAXParserFactoryfac=SAXParserFactory.newInstance();
- XMLReaderreader=fac.newSAXParser().getXMLReader();
- SaxParserServicesaxHandler=newSaxParserService();
- reader.setContentHandler(saxHandler);
- reader.parse(newInputSource(input));
- students=saxHandler.getParseData();
- //测试DOM解析方式
- voidtestDom() students=DomParseService.getPersonsByParseXml(input);
- for(Studentstu: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文件中添加如下两行
copy
instrumentationandroid:name="android.test.InstrumentationTestRunner"android:targetPackage="com.and.pull">instrumentation>
uses-libraryandroid:name="android.test.runner"/>
注意添加位置:
第一句跟application节点同级。
第二句跟activity同级。
上面介绍的方法是手动代码添加,下面介绍一种图形化的方式,只需要点击鼠标就可以搞定。
打开AndroidManifest.xml文件
点击Add...
这样,use-library就添加好了
同样的方法添加instrumentation属性
注意Target package后面的内容:com.and.xml包
整个工程目录结构如图:
然后查看一下AndroidManifest.xml的内容,已经包含了刚才添加的那两句了:
copy
xmlversion="1.0"encoding="utf-8"manifestxmlns:android="http://schemas.android.com/apk/res/android"
- package="com.and.xml"
- android:versionCode="1"
- android:versionName="1.0"uses-sdkandroid:minSdkVersion="7"/>
- instrumentationandroid:targetPackage="com.and.test"android:name="android.test.InstrumentationTestRunner"application
- android:icon="@drawable/ic_launcher"
- android:label="@string/app_name"activity
- android:label="@string/app_name"
- android:name="com.and.xml.MainActivity"intent-filteractionandroid:name="android.intent.action.MAIN"categoryandroid:name="android.intent.category.LAUNCHER"intent-filteractivity/>
- applicationmanifest>
OK,然后继续之前的操作“鼠标选中testPull方法名——>右键——>Run As——>Anroid JUnit Test”
如果出现类似这样的页面,就表示测试用例建立成功,并且测试方法通过
左上角的绿色条,表示测试方法通过,右下角的调试日志输出,通过判断可以知道解析成功。
第二种:SAX解析
SaxParserService.java
copy
importjava.util.ArrayList;
- importorg.xml.sax.Attributes;
- importorg.xml.sax.SAXException;
- importorg.xml.sax.helpers.DefaultHandler;
- /**
- *SAX解析示例
- classSaxParserServiceextendsDefaultHandler{
- Stringtag="";
- voidcharacters(char[]ch,153); font-weight:bold; background-color:inherit">intstart,153); font-weight:bold; background-color:inherit">intlength)throwsSAXException{
- super.characters(ch,start,length);
- Stringstr=newString(ch,153); font-weight:bold; background-color:inherit">if(tag.equals("name")){
- stu.setName(str);
- }if(tag.equals("age")){
- stu.setAge(Integer.parseInt(str));
- if(tag.equals("score")){
- stu.setScore(Float.parseFloat(str));
- @Override
- voidendDocument()throwsSAXException{
- super.endDocument();
- voidendElement(Stringuri,StringlocalName,StringqName)
- super.endElement(uri,localName,qName);
- if(localName.equals("student")&&stu!=null){
- data.add(stu);
- tag="";
- voidstartDocument()super.startDocument();
- voidstartElement(Stringuri,StringqName,
- Attributesattributes)super.startElement(uri,qName,attributes);
- tag=localName;
- if(localName.equals("student")){
- newStudent();
- if(attributes.getValue(0)!= stu.setId(Integer.parseInt(attributes.getValue(0)));
- publicList<Student>getParseData(){
- 注意一定要继承自DefaultHandler,然复写里面的方法,这些方法名字根据字面意思很容易理解它的作用。
然后通过上面的的测试文件,按照类似的方法测试一下testSAX()方法,如果出现绿条和日志输出的话,表明解析业务逻辑成功。
第三种:DOM解析
DomParseService.java
copy
importjavax.xml.parsers.DocumentBuilder;
- importjavax.xml.parsers.DocumentBuilderFactory;
- importorg.w3c.dom.Document;
- importorg.w3c.dom.Element;
- importorg.w3c.dom.Node;
- importorg.w3c.dom.NodeList;
- *DOM解析示例
- *DOM解析器在解析XML文档时,会把文档中的所有元素,按照其出现的层次关系,解析成一个个Node对象(节点)。Node对象提供了一系列常量来代表结点的类型
- *,当开发人员获得某个Node类型后,就可以把Node节点转换成相应节点对象(Node的子类对象),以便于调用其特有的方法。
- *Node对象提供了相应的方法去获得它的父结点或子结点。编程人员通过这些方法就可以读取整个XML文档的内容、或添加、修改、删除XML文档的内容.
- *缺点:一次性的完全加载整个xml文件,需要消耗大量的内存。
- */
- classDomParseService{
- staticList<Student>getPersonsByParseXml(InputStreamis) List<Student>persons= DocumentBuilderFactoryfactory=DocumentBuilderFactory.newInstance();
- try{
- DocumentBuilderbuilder=factory.newDocumentBuilder();
- Documentdocument=builder.parse(is);
- Elementroot=document.getDocumentElement();
- NodeListitems=root.getElementsByTagName("student");
- for(inti=0;i<items.getLength();i++){
- StudentStudent= ElementpersonNode=(Element)items.item(i);
- Student.setId(newInteger(personNode.getAttribute("id")));
- //获取person节点下的所有子节点(标签之间的空白节点和name/age元素)
- NodeListchildsNodes=personNode.getChildNodes();
- intj=0;j<childsNodes.getLength();j++){
- Nodenode=(Node)childsNodes.item(j);
- if(node.getNodeType()==Node.ELEMENT_NODE){
- ElementchildNode=(Element)node;
- //判断是否name元素
- if("name".equals(childNode.getNodeName())){
- //获取name元素下Text节点,然后从Text节点获取数据
- Student.setName(childNode.getFirstChild().getNodeValue());
- if("age".equals(childNode.getNodeName())){
- Student.setAge(newShort(childNode.getFirstChild().getNodeValue()));
- if("score".equals(childNode.getNodeName())){
- Student.setScore(Float.parseFloat(childNode.getFirstChild().getNodeValue()));
- persons.add(Student);
- is.close();
- catch(Exceptione){
- e.printStackTrace();
- returnpersons;
- }
类似的测试方法。。。
至此,三种解析方式全部完成了,如果分别测试这三种方法的时候,一路绿条的话,那么恭喜,解析业务逻辑成功。否则,可能还有哪里有问题,请仔细检查。
对比这三种解析方式,我个人认为PULL和SAX解析方式类似,都是事件触发型的,就是当解析到某个节点的时候触发相应的事件。说明一下DOM解析,会把文档中的所有元素,按照其出现的层次关系,解析成一个个Node对象(节点),可见它会有点占内存,但是如果待解析的xml文件相对较小的话,使用DOM解析 优点还是很明确的。 (编辑:李大同)
【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!
|