XML数据解析
XML解析<apps> <app> <id>1</id> <name>GoogleMaps</name> <version>1.0</version> </app> <app> <id>2</id> <name>Chrome</name> <version>2.1</version> </app> <app> <id>3</id> <name>GooglePlay</name> <version>2.3</version> </app> </apps> Pull解析方式解析XML格式的数据其实也有挺多种方式的,本节中我们学习比较常用的两种,Pull解析和SAX解析。那么简单起见,这里仍然是在NetworkTest项目的基础上继续开发,这样我们就可以重用之前网络通信部分的代码,从而把工作的重心放在XML数据解析上。 既然XML格式的数据已经提供好了,现在要做的就是从中解析出我们想要得到的那部分内容。修改MainActivity中的代码,如下所示: publicclassMainActivityextendsActivityimplementsOnClickListener{ …… privatevoidsendRequestWithHttpClient(){ newThread(newRunnable(){ @Override publicvoidrun(){ try{ HttpClienthttpClient=newDefaultHttpClient //指定访问的服务器地址是电脑本机 HttpGethttpGet=newHttpGet("http://10.0.2.2/get_data.xml"); HttpResponsehttpResponse=httpClient.execute(httpGet); if(httpResponse.getStatusLine().getStatusCode()==200){ //请求和响应都成功了 HttpEntityentity=httpResponse.getEntity(); Stringresponse=EntityUtils.toString(entity,"utf-8"); parseXMLWithPull(response); } }catch(Exceptione){ e.printStackTrace(); } } }).start(); } privatevoidparseXMLWithPull(StringxmlData){ try{ XmlPullParserFactoryfactory=XmlPullParserFactory.newInstance(); XmlPullParserxmlPullParser=factory.newPullParser(); xmlPullParser.setInput(newStringReader(xmlData)); inteventType=xmlPullParser.getEventType(); Stringid=""; Stringname=""; Stringversion=""; while(eventType!=XmlPullParser.END_DOCUMENT){ StringnodeName=xmlPullParser.getName(); switch(eventType){ //开始解析某个结点 caseXmlPullParser.START_TAG:{ if("id".equals(nodeName)){ id=xmlPullParser.nextText(); }elseif("name".equals(nodeName)){ name=xmlPullParser.nextText(); }elseif("version".equals(nodeName)){ version=xmlPullParser.nextText(); } break; } //完成解析某个结点 caseXmlPullParser.END_TAG:{ if("app".equals(nodeName)){ Log.d("MainActivity","idis"+id); Log.d("MainActivity","nameis"+name); Log.d("MainActivity","versionis"+version); } break; } default: break; } eventType=xmlPullParser.next(); } }catch(Exceptione){ e.printStackTrace(); } } } 可以看到,这里首先是将HTTP请求的地址改成了http://10.0.2.2/get_data.xml,10.0.2.2 对于模拟器来说就是电脑本机的IP地址。在得到了服务器返回的数据后,我们并不再去发送一条消息,而是调用了parseXMLWithPull()方法来解析服务器返回的数据。 下面就来仔细看下parseXMLWithPull()方法中的代码吧。这里首先要获取到一个 XmlPullParserFactory的实例,并借助这个实例得到XmlPullParser对象,然后调用XmlPullParser的setInput()方法将服务器返回的XML数据设置进去就可以开始解析了。解析的过程也是非常简单,通过getEventType()可以得到当前的解析事件,然后在一个while循环中不断地进行解析,如果当前的解析事件不等于XmlPullParser.END_DOCUMENT,说明解析工作还没完成,调用next()方法后可以获取下一个解析事件。 在while循环中,我们通过getName()方法得到当前结点的名字,如果发现结点名等于id、name或version,就调用nextText()方法来获取结点内具体的内容,每当解析完一个app结点后就将获取到的内容打印出来。 SAX解析方式 Pull解析方式虽然非常的好用,但它并不是我们唯一的选择。SAX解析也是一种特别常用的XML解析方式,虽然它的用法比Pull解析要复杂一些,但在语义方面会更加的清楚。 通常情况下我们都会新建一个类继承自DefaultHandler,并重写父类的五个方法,如下 所示: publicclassMyHandlerextendsDefaultHandler{ @Override publicvoidstartDocument()throwsSAXException{ } @Override publicvoidstartElement(Stringuri,StringlocalName,StringqName,Attributesattributes)throwsSAXException{ } @Override publicvoidcharacters(char[]ch,intstart,intlength)throwsSAXException{ } @Override publicvoidendElement(Stringuri,StringqName)throwsSAXException{ } @Override publicvoidendDocument()throwsSAXException{ } } 这五个方法一看就很清楚吧?startDocument()方法会在开始XML解析的时候调用, startElement()方法会在开始解析某个结点的时候调用,characters()方法会在获取结点中内容的时候调用,endElement()方法会在完成解析某个结点的时候调用,endDocument()方法会在完成整个XML解析的时候调用。其中,startElement()、characters()和endElement()这三个方法是有参数的,从XML中解析出的数据就会以参数的形式传入到这些方法中。需要注意的是,在获取结点中的内容时,characters()方法可能会被调用多次,一些换行符也被当作内容解析出来,我们需要针对这种情况在代码中做好控制。 那么下面就让我们尝试用SAX解析的方式来实现和上一小节中同样的功能吧。新建一个ContentHandler类继承自DefaultHandler,并重写父类的五个方法,如下所示: publicclassContentHandlerextendsDefaultHandler{ privateStringnodeName; privateStringBuilderid; privateStringBuildername; privateStringBuilderversion; @Override publicvoidstartDocument()throwsSAXException{ id=newStringBuilder(); name=newStringBuilder(); version=newStringBuilder(); } @Override publicvoidstartElement(Stringuri,Attributesattributes)throwsSAXException{ //记录当前结点名 nodeName=localName; } @Override publicvoidcharacters(char[]ch,intlength)throwsSAXException{ //根据当前的结点名判断将内容添加到哪一个StringBuilder对象中 if("id".equals(nodeName)){ id.append(ch,start,length); }elseif("name".equals(nodeName)){ name.append(ch,length); }elseif("version".equals(nodeName)){ version.append(ch,length); } } @Override publicvoidendElement(Stringuri,StringqName)throwsSAXException{ if("app".equals(localName)){ Log.d("ContentHandler","idis"+id.toString().trim()); Log.d("ContentHandler","nameis"+name.toString().trim()); Log.d("ContentHandler","versionis"+version.toString().trim()); //最后要将StringBuilder清空掉 id.setLength(0); name.setLength(0); version.setLength(0); } } @Override publicvoidendDocument()throwsSAXException{ } } 可以看到,我们首先给id、name和version结点分别定义了一个StringBuilder对象,并在startDocument()方法里对它们进行了初始化。每当开始解析某个结点的时候,startElement()方法就会得到调用,其中localName参数记录着当前结点的名字,这里我们把它记录下来。 接着在解析结点中具体内容的时候就会调用characters()方法,我们会根据当前的结点名进行判断,将解析出的内容添加到哪一个StringBuilder对象中。最后在endElement()方法中进行判断,如果app结点已经解析完成,就打印出id、name和version的内容。需要注意的是,目前id、name和version中都可能是包括回车或换行符的,因此在打印之前我们还需要调用一下trim()方法,并且打印完成后还要将StringBuilder的内容清空掉,不然的话会影响下一次内容的读取。 接下来的工作就非常简单了,修改MainActivity中的代码,如下所示: publicclassMainActivityextendsActivityimplementsOnClickListener{ …… privatevoidsendRequestWithHttpClient(){ newThread(newRunnable(){ @Override publicvoidrun(){ try{ HttpClienthttpClient=newDefaultHttpClient(); //指定访问的服务器地址是电脑本机 HttpGethttpGet=newHttpGet("http://10.0.2.2:8080/ get_data.xml"); HttpResponsehttpResponse=httpClient.execute(httpGet); if(httpResponse.getStatusLine().getStatusCode()==200){ //请求和响应都成功了 HttpEntityentity=httpResponse.getEntity(); Stringresponse=EntityUtils.toString(entity,"utf-8"); parseXMLWithSAX(response); } }catch(Exceptione){ e.printStackTrace(); } } }).start(); } …… privatevoidparseXMLWithSAX(StringxmlData){ try{ SAXParserFactoryfactory=SAXParserFactory.newInstance(); XMLReaderxmlReader=factory.newSAXParser().getXMLReader(); ContentHandlerhandler=newContentHandler(); //将ContentHandler的实例设置到XMLReader中 xmlReader.setContentHandler(handler); //开始执行解析 xmlReader.parse(newInputSource(newStringReader(xmlData))); }catch(Exceptione){ e.printStackTrace(); } } } 在得到了服务器返回的数据后,我们这次去调用parseXMLWithSAX()方法来解析XML数据。parseXMLWithSAX()方法中先是创建了一个SAXParserFactory的对象,然后再获取到XMLReader对象,接着将我们编写的ContentHandler的实例设置到XMLReader中,最后调用parse()方法开始执行解析就好了。现在重新运行一下程序,点击SendRequest按钮后观察LogCat中的打印日志,你会看到和图10.7中一样的结果。除了Pull解析和SAX解析之外,其实还有一种DOM解析方式也算挺常用的,不过这里我们就不再展开进行讲解了,感兴趣的话你可以自己去查阅一下相关资料。 (编辑:李大同) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |