XML解析(一),SAX解析XML
一、概述?SAX,全称Simple API for XML,是一种以事件驱动的XMl API,是XML解析的一种新的替代方法,解析XML常用的还有DOM解析,PULL解析(Android特有),SAX与DOM不同的是它边扫描边解析,自顶向下依次解析,由于边扫描边解析,所以它解析XML具有速度快,占用内存少的优点,对于Android等CPU资源宝贵的移动平台来说是一个巨大的优势。
二、SAX解析的步骤解析步骤很简单,可分为以下四个步骤
知道了SAX解析的优缺点和解析步骤,下面我们通过一个简单的Demo学习一下SAX解析XML 三、SAX解析实战新建一个Android工程叫SaxParseXmlDemo,将sax.jar下载放到工程的lib下面并添加到构建路径中,为了方便,我先将工程的目录结构列一下: 1、新建一个xml文件叫users.xml <?xml version="1.0" encoding="UTF-8"?>
<users>
<user id="1">
<name>毕向东</name>
<password>bxd123</password>
</user>
<user id="2">
<name>韩顺平</name>
<password>hsp123</password>
</user>
<user id="3">
<name>马士兵</name>
<password>msb123</password>
</user>
</users>
2、新建一个JavaBean package com.example.saxparsexmldemo;
public class User {
private long id;
private String name;
private String password;
public long getId() {
return id;
}
public void setId(long id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String getPassword() {
return password;
}
public void setPassword(String password) {
this.password = password;
}
}
3、新建一个类XmlParseHandler.java,该类需要继承
?重写startDocument(),我们在这里初始化User集合,该集合用来存放解析出来的user Log.e("startDocument","startDocument()");
users = new ArrayList<User>();
?重写startElement,在 Log.e("startElement",localName + "-startElement()");
if ("user".equals(localName)) { // 是一个用户
for (int i = 0; i < attributes.getLength(); i++) {
Log.e("attributes","attribute_name:" + attributes.getLocalName(i)
+ " attribute_value:" + attributes.getValue(i));
user = new User();
if("id".equals(attributes.getLocalName(i))){
user.setId(Long.parseLong(attributes.getValue(i)));
}
}
}
currentTag = localName; // 把当前标签记录下来
?重写characters,在 String value = new String(ch,start,length); // 将当前TextNode转换为String
Log.e("characters",value+"");
if("name".equals(currentTag)){ // 当前标签为name标签,该标签无子标签,直接将上面获取到的标签的值封装到当前User对象中
// 该节点为name节点
user.setName(value);
}else if("password".equals(currentTag)){ // 当前标签为password标签,该标签无子标签,直接将上面获取到的标签的值封装到当前User对象中
// 该节点为password节点
user.setPassword(value);
}
?重写endElement,在这个方法中判断当前是否是user标签的结束,如果是user标签结束,则这个user信息解析结束,并将当前的User对象和当前的标签重置 Log.e("endElement",localName + "-endElement()");
if("user".equals(localName)){
users.add(user);
user = null;
}
currentTag = null;
?重写endDocument,这里直接给个空实现,我们只需观察Log输出 Log.e("endDocument","-endDocument()");
XmlParseHandler.java完整代码: package com.example.saxparsexmldemo;
import java.util.ArrayList;
import java.util.List;
import org.xml.sax.Attributes;
import org.xml.sax.SAXException;
import org.xml.sax.helpers.DefaultHandler;
import android.util.Log;
public class XmlParseHandler extends DefaultHandler {
private List<User> users;
private String currentTag; // 记录当前解析到的节点名称
private User user; // 记录当前的user
/** * 文档解析结束后调用 */
@Override
public void endDocument() throws SAXException {
super.endDocument();
Log.e("endDocument","-endDocument()");
}
/** * 节点解析结束后调用 * @param uri : 命名空间的uri * @param localName : 标签的名称 * @param qName : 带命名空间的标签名称 */
@Override
public void endElement(String uri,String qName)
throws SAXException {
super.endElement(uri,localName,qName);
Log.e("endElement",localName + "-endElement()");
if("user".equals(localName)){
users.add(user);
user = null;
}
currentTag = null;
}
/** * 文档解析开始调用 */
@Override
public void startDocument() throws SAXException {
super.startDocument();
Log.e("startDocument","startDocument()");
users = new ArrayList<User>();
}
/** * 节点解析开始调用 * @param uri : 命名空间的uri * @param localName : 标签的名称 * @param qName : 带命名空间的标签名称 */
@Override
public void startElement(String uri,Attributes attributes) throws SAXException {
super.startElement(uri,qName,attributes);
Log.e("startElement","attribute_name:" + attributes.getLocalName(i)
+ " attribute_value:" + attributes.getValue(i));
user = new User();
if("id".equals(attributes.getLocalName(i))){
user.setId(Long.parseLong(attributes.getValue(i)));
}
}
}
currentTag = localName; // 把当前标签记录下来
}
@Override
public void characters(char[] ch,int start,int length)
throws SAXException {
super.characters(ch,length);
String value = new String(ch,value+"");
if("name".equals(currentTag)){ // 当前标签为name标签,该标签无子标签,直接将上面获取到的标签的值封装到当前User对象中
// 该节点为name节点
user.setName(value);
}else if("password".equals(currentTag)){ // 当前标签为password标签,该标签无子标签,直接将上面获取到的标签的值封装到当前User对象中
// 该节点为password节点
user.setPassword(value);
}
}
public List<User> getUsers() {
return users;
}
}
4、新建一个类XmlParseUtils.java,写一个方法进行xml解析 public static List<User> getUsers() throws ParserConfigurationException,SAXException,IOException {
// 加载文件返回文件的输入流
InputStream is = XmlParseUtils.class.getClassLoader().getResourceAsStream("users.xml");
XmlParseHandler handler = new XmlParseHandler();
// 1. 得到SAX解析工厂
SAXParserFactory saxParserFactory = SAXParserFactory.newInstance();
// 2. 让工厂生产一个sax解析器
SAXParser newSAXParser = saxParserFactory.newSAXParser();
// 3. 传入输入流和handler,解析
newSAXParser.parse(is,handler);
is.close();
return handler.getUsers();
}
?其中第三步也可以通过XMLReader来完成 XMLReader xmlReader = newSAXParser.getXMLReader();
InputSource input = new InputSource(is);
xmlReader.parse(input );
到这里SAX解析XML的代码完成了,OK,万事具备,只欠测试了 这里我们用Android的单元测试,只需完成调用XmlParseUtils的getUsers()方法测试输出即可 Android的单元测试(需要连接模拟器或者手机)
public void testSAXgetUsers() throws Exception{
List<User> users = XmlParseUtils.getUsers();
for(User user : users){
Log.e(TAG,"name:"+user.getName());
Log.e(TAG,"password:"+user.getPassword());
}
}
?OK,右键该类的testSAXgetUsers,Run As Android JUnit Test,我们可以看到XML解析成功 通过上面的这个小Demo的完成,我们可以看到SAX解析代码不多也不难理解,关键是Handler的几个方法的使用,即遇到什么符号会触发什么事件,以及触发过程,掌握了SAX的事件触发,那么就掌握了SAX解析XML,下面我们来分析一下SAX解析的原理或流程 四、SAX解析XML原理在分析SAX解析原理之前,我们先看一下上面的demo的日志输出 以users.xml的解析过程为例,结合上面的日志输出,我们可以分析出SAX解析的流程 1、xml解析开始, 下面画一个图让我们进一步理解SAX解析XML的原理: 对上图做个简单说明,当当前标签有子标签的时候,该标签先触发 OK,SAX解析原理分析完了,最后用一句话描述SAX解析过程 startDocument-startElement-characters-endElement-endDocument 总结,SAX解析XML具有解析速度快,占用内存少,对于Android等移动设备来说有巨大的优势,深入了解SAX的事件触发机制是掌握SAX解析的关键,掌握了SAX的事件触发就掌握了SAX解析XML上面这篇文章由于个人理解,如果有理解错的地方,欢迎大家指出,与君共勉,一起进步。 Demo下载地址:http://download.csdn.net/detail/ydxlt/9328309下篇文章:【XML解析(二)】DOM解析XML (编辑:李大同) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |