(以下文章基本照抄郭霖大神的《第一行代码》)
在Android之 解析XML文件(1)—— Pull解析 中我们讲了Pull方式解析XML文件。今天讲另外一种方式,SAX解析XML文件。
首先还是先看代码。
一、 SAX解析参考代码
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();
}
}
这里我们看到SAX解析看起来要比Pull解析简洁明了很多。
二、 相关类简介
1、 SAXParserFactory
SAXParserFactory 与 XmlPullParserFactory类相似。也是提供SAXParser 实例的一个工厂。
下面是Android API中对 SAXParserFactory类的定义。
下面是SAXParserFactory中定义的方法。代码中使用到的 newInstance() 与 newSAXParser() 分别用于产生一个SAXParserFactory实例和产生一个SAXParser实例。
2、XMLReader
XMLReader是一个接口。通过它的setContentHandler()方法,可以设置解析事件的处理handler,通过parse()方法可以开始解析。
XMLReader的实例是通过SAXParser类的 getXMLReader()方法来获取的。
3、ContentHandler
上诉代码中ContentHandler类是我们自己写的类,继承自DefaultHandler类。DefaultHandler类是SAX2事件处理的默认基础类。
它提供了在四个核心的SAX处理类中的所有回调的默认实现。我们可以继承该类,在对应的方法中重写我们的处理逻辑。
下面是Android API中对DefaultHandler类的描述。
我们继承的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{
}
}
可以看到在ContentHandler这个处理类中对xml数据的处理方式与Pull解析是类似的。所以SAX解析只不过是将解析用到的方法封装起来,代码书写的时候逻辑更为清晰。
综上,使用SAX解析XML数据我们需要做以下几步:
1、使用SAXParserFactory 类中的newInstance()方法获取SAXParserFactory 类实例。
2、通过SAXParserFactory 类实例的newSAXParser()方法获取SAXParser实例
3、通过SAXParser实例的getXMLReader()获取XMLReader的实例
4、XMLReader实例调用setContentHandler(ContentHandler contentHandler)方法设置解析所需的处理事件
5、自定义一个类继承自 DefaultHandler,重写我们需要的方法(这些方法里面是我们用于处理XML数据的逻辑)。
6、最后调用XMLReader的parse()方法解析数据。