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解析方式也算挺常用的,不过这里我们就不再展开进行讲解了,感兴趣的话你可以自己去查阅一下相关资料。
原文链接:https://www.f2er.com/xml/295842.html