Android 解析XML格式数据

解析XML格式数据

Pull解析方式

解析XML格式的数据其实也有挺多种方式的,本节中我们学习比较常用的两种,Pull解析和SAX解析。那么简单起见,这里仍然是在NetworkTest项目的基础上继续开发,这样我们就可以重用之前网络通信部分的代码,从而把工作的重心放在XML数据解析上。

既然XML格式的数据已经提供好了,现在要做的就是从中解析出我们想要得到的那部分内容。修改MainActivity中的代码,如下所示:

public class MainActivity extends Activity implements OnClickListener {

……

private void sendRequestWithHttpClient() {

new Thread(new Runnable() {

@Override

public void run() {

try {

HttpClient httpClient = new DefaultHttpClient();

// 指定访问的服务器地址是电脑本机

HttpGet httpGet = new HttpGet("http://10.0.2.2/get_data.xml");

HttpResponse httpResponse = httpClient.execute(httpGet);

if (httpResponse.getStatusLine().getStatusCode() == 200) {

// 请求和响应都成功了

HttpEntity entity = httpResponse.getEntity();

String response = EntityUtils.toString(entity, "utf-8");

parseXMLWithPull(response);

}

} catch (Exception e) {

e.printStackTrace();

}

}

}).start();

}


private void parseXMLWithPull(String xmlData) {

try {

XmlPullParserFactory factory = XmlPullParserFactory.newInstance();

XmlPullParser xmlPullParser = factory.newPullParser();

xmlPullParser.setInput(new StringReader(xmlData));

int eventType = xmlPullParser.getEventType();

String id = "";

String name = "";

String version = "";

while (eventType != XmlPullParser.END_DOCUMENT) {

String nodeName = xmlPullParser.getName();

switch (eventType) {

// 开始解析某个结点

case XmlPullParser.START_TAG: {

if ("id".equals(nodeName)) {

id = xmlPullParser.nextText();

} else if ("name".equals(nodeName)) {

name = xmlPullParser.nextText();

} else if ("version".equals(nodeName)) {

version = xmlPullParser.nextText();

}

break;

}

// 完成解析某个结点

case XmlPullParser.END_TAG: {

if ("app".equals(nodeName)) {

Log.d("MainActivity", "id is " + id);

Log.d("MainActivity", "name is " + name);

Log.d("MainActivity", "version is " + version);

}

break;

}

default:

break;

}

eventType = xmlPullParser.next();

}

} catch (Exception e) {

e.printStackTrace();

}

}

}

可以看到,这里首先是将HTTP请求的地址改成了http://10.0.2.2/get_data.xml10.0.2.2对于模拟器来说就是电脑本机的IP地址。在得到了服务器返回的数据后,我们并不再去发送一条消息,而是调用了parseXMLWithPull()方法来解析服务器返回的数据。

下面就来仔细看下parseXMLWithPull()方法中的代码吧。这里首先要获取到一个XmlPullParserFactory的实例,并借助这个实例得到XmlPullParser对象,然后调用XmlPullParsersetInput()方法将服务器返回的XML数据设置进去就可以开始解析了。解析的过程也是非常简单,通过getEventType()可以得到当前的解析事件,然后在一个while循环中不断地进行解析,如果当前的解析事件不等于XmlPullParser.END_DOCUMENT,说明解析工作还没完成,调用next()方法后可以获取下一个解析事件。

while循环中,我们通过getName()方法得到当前结点的名字,如果发现结点名等于idnameversion,就调用nextText()方法来获取结点内具体的内容,每当解析完一个app结点后就将获取到的内容打印出来。

好了,整体的过程就是这么简单,下面就让我们来测试一下吧。运行NetworkTest项目,然后点击Send Request按钮,观察LogCat中的打印日志,如图所示。


可以看到,我们已经将XML数据中的指定内容成功解析出来了。

SAX解析方式

Pull解析方式虽然非常的好用,但它并不是我们唯一的选择。SAX解析也是一种特别常用的XML解析方式,虽然它的用法比Pull解析要复杂一些,但在语义方面会更加的清楚。

通常情况下我们都会新建一个类继承自DefaultHandler,并重写父类的五个方法,如下所示:

public class MyHandler extends DefaultHandler {

@Override

public void startDocument() throws SAXException {

}


@Override

public void startElement(String uri, String localName, String qName, Attributes attributes) throws SAXException {

}


@Override

public void characters(char[] ch, int start, int length) throws SAXException {

}


@Override

public void endElement(String uri, String localName, String qName) throws SAXException {

}


@Override

public void endDocument() throws SAXException {

}


}

这五个方法一看就很清楚吧?startDocument()方法会在开始XML解析的时候调用,startElement()方法会在开始解析某个结点的时候调用,characters()方法会在获取结点中内容的时候调用,endElement()方法会在完成解析某个结点的时候调用,endDocument()方法会在完成整个XML解析的时候调用。其中,startElement()characters()endElement()这三个方法是有参数的,从XML中解析出的数据就会以参数的形式传入到这些方法中。需要注意的是,在获取结点中的内容时,characters()方法可能会被调用多次,一些换行符也被当作内容解析出来,我们需要针对这种情况在代码中做好控制。

那么下面就让我们尝试用SAX解析的方式来实现和上一小节中同样的功能吧。新建一个ContentHandler类继承自DefaultHandler,并重写父类的五个方法,如下所示:

public class ContentHandler extends DefaultHandler {


private String nodeName;


private StringBuilder id;


private StringBuilder name;


private StringBuilder version;


@Override

public void startDocument() throws SAXException {

id = new StringBuilder();

name = new StringBuilder();

version = new StringBuilder();

}


@Override

public void startElement(String uri, String localName, String qName, Attributes attributes) throws SAXException {

// 记录当前结点名

nodeName = localName;

}


@Override

public void characters(char[] ch, int start, int length) throws SAXException {

// 根据当前的结点名判断将内容添加到哪一个StringBuilder对象中

if ("id".equals(nodeName)) {

id.append(ch, start, length);

} else if ("name".equals(nodeName)) {

name.append(ch, start, length);

} else if ("version".equals(nodeName)) {

version.append(ch, start, length);

}

}


@Override

public void endElement(String uri, String localName, String qName) throws SAXException {

if ("app".equals(localName)) {

Log.d("ContentHandler", "id is " + id.toString().trim());

Log.d("ContentHandler", "name is " + name.toString().trim());

Log.d("ContentHandler", "version is " + version.toString().trim());

// 最后要将StringBuilder清空掉

id.setLength(0);

name.setLength(0);

version.setLength(0);

}

}


@Override

public void endDocument() throws SAXException {

}


}

可以看到,我们首先给idnameversion结点分别定义了一个StringBuilder对象,并在startDocument()方法里对它们进行了初始化。每当开始解析某个结点的时候,startElement()方法就会得到调用,其中localName参数记录着当前结点的名字,这里我们把它记录下来。接着在解析结点中具体内容的时候就会调用characters()方法,我们会根据当前的结点名进行判断,将解析出的内容添加到哪一个StringBuilder对象中。最后在endElement()方法中进行判断,如果app结点已经解析完成,就打印出idnameversion的内容。需要注意的是,目前idnameversion中都可能是包括回车或换行符的,因此在打印之前我们还需要调用一下trim()方法,并且打印完成后还要将StringBuilder的内容清空掉,不然的话会影响下一次内容的读取。

接下来的工作就非常简单了,修改MainActivity中的代码,如下所示:

public class MainActivity extends Activity implements OnClickListener {

……

private void sendRequestWithHttpClient() {

new Thread(new Runnable() {

@Override

public void run() {

try {

HttpClient httpClient = new DefaultHttpClient();

// 指定访问的服务器地址是电脑本机

HttpGet httpGet = new HttpGet("http://10.0.2.2:8080/ get_data.xml");

HttpResponse httpResponse = httpClient.execute(httpGet);

if (httpResponse.getStatusLine().getStatusCode() == 200) {

// 请求和响应都成功了

HttpEntity entity = httpResponse.getEntity();

String response = EntityUtils.toString(entity, "utf-8");

parseXMLWithSAX(response);

}

} catch (Exception e) {

e.printStackTrace();

}

}

}).start();

}

……

private void parseXMLWithSAX(String xmlData) {

try {

SAXParserFactory factory = SAXParserFactory.newInstance();

XMLReader xmlReader = factory.newSAXParser().getXMLReader();

ContentHandler handler = new ContentHandler();

// ContentHandler的实例设置到XMLReader

xmlReader.setContentHandler(handler);

// 开始执行解析

xmlReader.parse(new InputSource(new StringReader(xmlData)));

} catch (Exception e) {

e.printStackTrace();

}

}

}

在得到了服务器返回的数据后,我们这次去调用parseXMLWithSAX()方法来解析XML数据。parseXMLWithSAX()方法中先是创建了一个SAXParserFactory的对象,然后再获取到XMLReader对象,接着将我们编写的ContentHandler的实例设置到XMLReader中,最后调用parse()方法开始执行解析就好了。

现在重新运行一下程序,点击Send Request按钮后观察LogCat中的打印日志,你会看到和图10.7中一样的结果。

除了Pull解析和SAX解析之外,其实还有一种DOM解析方式也算挺常用的,不过这里我们就不再展开进行讲解了,感兴趣的话你可以自己去查阅一下相关资料。












  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值