加入收藏 | 设为首页 | 会员中心 | 我要投稿 李大同 (https://www.lidatong.com.cn/)- 科技、建站、经验、云计算、5G、大数据,站长网!
当前位置: 首页 > 百科 > 正文

XML数据解析

发布时间:2020-12-16 06:02:11 所属栏目:百科 来源:网络整理
导读:XML 解析 apps app id1/id nameGoogleMaps/name version1.0/version /app app id2/id nameChrome/name version2.1/version /app app id3/id nameGooglePlay/name version2.3/version /app /apps Pull解析方式 解析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解析方式也算挺常用的,不过这里我们就不再展开进行讲解了,感兴趣的话你可以自己去查阅一下相关资料。

(编辑:李大同)

【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!

    推荐文章
      热点阅读