SAX解析XML文件
2015-11-22 23:11
495 查看
SAX是一个解析速度快并且占用内存少的xml解析器,非常适合用于Android等移动设备。
SAX解析XML文件采用的是事件驱动,也就是说,它并不需要解析完整个文档,在按内容顺序解析文档的过程中,SAX会判断当前读到的字符是否合法XML语法中的某部分,如果符合就会触发事件。所谓事件,其实就是一些回调(callback)方法,这些方法(事件)定义在ContentHandler接口。下面是一些ContentHandler接口常用的方法:
startDocument()
当遇到文档的开头的时候,调用这个方法,可以在其中做一些预处理的工作。
endDocument()
和上面的方法相对应,当文档结束的时候,调用这个方法,可以在其中做一些善后的工作。
startElement(StringnamespaceURI, String localName, String qName, Attributes atts)
当读到一个开始标签的时候,会触发这个方法。namespaceURI就是命名空间,localName是不带命名空间前缀的标签名,qName是带命名空间前缀的标签名。通过atts可以得到所有的属性名和相应的值。要注意的是SAX中一个重要的特点就是它的流式处理,当遇到一个标签的时候,它并不会纪录下以前所碰到的标签,也就是说,在startElement()方法中,所有你所知道的信息,就是标签的名字和属性,至于标签的嵌套结构,上层标签的名字,是否有子元属等等其它与结构相关的信息,都是不得而知的,都需要你的程序来完成。这使得SAX在编程处理上没有DOM来得那么方便。
endElement(String uri, StringlocalName, String name)
这个方法和上面的方法相对应,在遇到结束标签的时候,调用这个方法。
characters(char[] ch, intstart, int length)
这个方法用来处理在XML文件中读到的内容,第一个参数用于存放文件的内容,后面两个参数是读到的字符串在这个数组中的起始位置和长度,使用newString(ch,start,length)就可以获取内容。
只要为SAX提供实现ContentHandler接口的类,那么该类就可以得到通知事件(实际上就是SAX调用了该类中的回调方法)。因为ContentHandler是一个接口,在使用的时候可能会有些不方便,因此,SAX还为其制定了一个Helper类:DefaultHandler,它实现了这个接口,但是其所有的方法体都为空,在实现的时候,你只需要继承这个类,然后重载相应的方法即可。使用SAX解析itcast.xml的代码如下:
public staticList<Person> readXML(InputStream inStream) {
try {
SAXParserFactory spf =SAXParserFactory.newInstance();
SAXParser saxParser = spf.newSAXParser();//创建解析器
//设置解析器的相关特性,http://xml.org/sax/features/namespaces= true
表示开启命名空间特性
saxParser.setProperty(“http://xml.org/sax/features/namespaces”,true);
XMLContentHandler handler = newXMLContentHandler();
saxParser.parse(inStream,handler);
inStream.close();
return handler.getPersons();
} catch (Exception e) {
e.printStackTrace();
}
return null;
}
SAX 支持已内置到JDK1.5中,你无需添加任何的jar文件。XMLContentHandler的代码:
import java.util.ArrayList;
import java.util.List;
import org.xml.sax.Attributes;
import org.xml.sax.SAXException;
importorg.xml.sax.helpers.DefaultHandler;
importcn.itcast.xml.domain.Person;
public class XMLContentHandlerextends DefaultHandler {
private List<Person>persons = null;
private Person currentPerson;
private String tagName = null;//当前解析的元素标签
public List<Person>getPersons() {
return persons;
}
/*
* 接收文档的开始的通知。
*/
@Override public voidstartDocument() throws SAXException {
persons = newArrayList<Person>();
}
/*
* 接收字符数据的通知。
*/
@Override public voidcharacters(char[] ch, int start, int length) throws SAXException {
if(tagName!=null){
String data = new String(ch,start, length);
if(tagName.equals(“name”)){
this.currentPerson.setName(data);
}elseif(tagName.equals(“age”)){
this.currentPerson.setAge(Short.parseShort(data));
}
}
}
/*
* 接收元素开始的通知。
* 参数意义如下:
* namespaceURI:元素的命名空间
* localName :元素的本地名称(不带前缀)
* qName :元素的限定名(带前缀)
* atts :元素的属性集合
*/
@Override public voidstartElement(String namespaceURI, String localName, String qName, Attributesatts) throws SAXException {
if(localName.equals(“person”)){
currentPerson = new Person();
currentPerson.setId(Integer.parseInt(atts.getValue(“id”)));
}
this.tagName = localName;
}
/*
* 接收文档的结尾的通知。
* 参数意义如下:
* uri :元素的命名空间
* localName :元素的本地名称(不带前缀)
* name :元素的限定名(带前缀)
*
*/
@Override public voidendElement(String uri, String localName, String name) throws SAXException {
if(localName.equals(“person”)){
persons.add(currentPerson);
currentPerson = null;
}
this.tagName = null;
}
}
SAX解析XML文件采用的是事件驱动,也就是说,它并不需要解析完整个文档,在按内容顺序解析文档的过程中,SAX会判断当前读到的字符是否合法XML语法中的某部分,如果符合就会触发事件。所谓事件,其实就是一些回调(callback)方法,这些方法(事件)定义在ContentHandler接口。下面是一些ContentHandler接口常用的方法:
startDocument()
当遇到文档的开头的时候,调用这个方法,可以在其中做一些预处理的工作。
endDocument()
和上面的方法相对应,当文档结束的时候,调用这个方法,可以在其中做一些善后的工作。
startElement(StringnamespaceURI, String localName, String qName, Attributes atts)
当读到一个开始标签的时候,会触发这个方法。namespaceURI就是命名空间,localName是不带命名空间前缀的标签名,qName是带命名空间前缀的标签名。通过atts可以得到所有的属性名和相应的值。要注意的是SAX中一个重要的特点就是它的流式处理,当遇到一个标签的时候,它并不会纪录下以前所碰到的标签,也就是说,在startElement()方法中,所有你所知道的信息,就是标签的名字和属性,至于标签的嵌套结构,上层标签的名字,是否有子元属等等其它与结构相关的信息,都是不得而知的,都需要你的程序来完成。这使得SAX在编程处理上没有DOM来得那么方便。
endElement(String uri, StringlocalName, String name)
这个方法和上面的方法相对应,在遇到结束标签的时候,调用这个方法。
characters(char[] ch, intstart, int length)
这个方法用来处理在XML文件中读到的内容,第一个参数用于存放文件的内容,后面两个参数是读到的字符串在这个数组中的起始位置和长度,使用newString(ch,start,length)就可以获取内容。
只要为SAX提供实现ContentHandler接口的类,那么该类就可以得到通知事件(实际上就是SAX调用了该类中的回调方法)。因为ContentHandler是一个接口,在使用的时候可能会有些不方便,因此,SAX还为其制定了一个Helper类:DefaultHandler,它实现了这个接口,但是其所有的方法体都为空,在实现的时候,你只需要继承这个类,然后重载相应的方法即可。使用SAX解析itcast.xml的代码如下:
public staticList<Person> readXML(InputStream inStream) {
try {
SAXParserFactory spf =SAXParserFactory.newInstance();
SAXParser saxParser = spf.newSAXParser();//创建解析器
//设置解析器的相关特性,http://xml.org/sax/features/namespaces= true
表示开启命名空间特性
saxParser.setProperty(“http://xml.org/sax/features/namespaces”,true);
XMLContentHandler handler = newXMLContentHandler();
saxParser.parse(inStream,handler);
inStream.close();
return handler.getPersons();
} catch (Exception e) {
e.printStackTrace();
}
return null;
}
SAX 支持已内置到JDK1.5中,你无需添加任何的jar文件。XMLContentHandler的代码:
import java.util.ArrayList;
import java.util.List;
import org.xml.sax.Attributes;
import org.xml.sax.SAXException;
importorg.xml.sax.helpers.DefaultHandler;
importcn.itcast.xml.domain.Person;
public class XMLContentHandlerextends DefaultHandler {
private List<Person>persons = null;
private Person currentPerson;
private String tagName = null;//当前解析的元素标签
public List<Person>getPersons() {
return persons;
}
/*
* 接收文档的开始的通知。
*/
@Override public voidstartDocument() throws SAXException {
persons = newArrayList<Person>();
}
/*
* 接收字符数据的通知。
*/
@Override public voidcharacters(char[] ch, int start, int length) throws SAXException {
if(tagName!=null){
String data = new String(ch,start, length);
if(tagName.equals(“name”)){
this.currentPerson.setName(data);
}elseif(tagName.equals(“age”)){
this.currentPerson.setAge(Short.parseShort(data));
}
}
}
/*
* 接收元素开始的通知。
* 参数意义如下:
* namespaceURI:元素的命名空间
* localName :元素的本地名称(不带前缀)
* qName :元素的限定名(带前缀)
* atts :元素的属性集合
*/
@Override public voidstartElement(String namespaceURI, String localName, String qName, Attributesatts) throws SAXException {
if(localName.equals(“person”)){
currentPerson = new Person();
currentPerson.setId(Integer.parseInt(atts.getValue(“id”)));
}
this.tagName = localName;
}
/*
* 接收文档的结尾的通知。
* 参数意义如下:
* uri :元素的命名空间
* localName :元素的本地名称(不带前缀)
* name :元素的限定名(带前缀)
*
*/
@Override public voidendElement(String uri, String localName, String name) throws SAXException {
if(localName.equals(“person”)){
persons.add(currentPerson);
currentPerson = null;
}
this.tagName = null;
}
}
相关文章推荐
- MongoDB——运维技术
- 20135328陈都信息安全系统设计基础第十二周学习总结
- redis单实例、多实例、主从同步实战
- JavaScript学习之路09_闭包
- RelativeLayout子控件的一些属性
- 1005-行列式
- AfxBeginThread和CreateThread具体区别
- javascript_DOM 编程艺术学习笔记(七)
- List、Set、Map的区别
- 平民架构的春天――UCloud数据方舟实战记
- 南大软院21天学霸养成计划—第6天
- SpringMVC之controller篇
- Json解析教程
- g++编写与使用动态链接库so(shared object)(附cmake生成动态库)
- draw2d 使用心得
- 学习笔记12-OC 内存管理
- NLog 錯誤小記
- MS-SQL数据库定时自动备份
- HTML总结
- Android进程