JAVA 解析 XML 通常有两种方式,DOM和SAX。DOM 虽然是 W3C 的标准,提供了标准的解析方式,但它的解析效率一直不尽如人意,因为使用DOM解析XML时,解析器读入整个文档并构建一个驻留内存的树结构(节点树),然后您的代码才可以使用 DOM 的标准接口来操作这个树结构。但大部分情况下我们只对文档的部分内容感兴趣,根本就不用先解析整个文档,并且从节点树的根节点来索引一些我们需要的数据也是非常耗时的。
SAX是一种XML解析的替代方法。相比于文档对象模型DOM,SAX 是读取和操作 XML 数据的更快速、更轻量的方
法。SAX 允许您在读取文档时处理它,从而不必等待整个文档被存储之后才采取操作。它不涉及 DOM 所必需的开销和概念跳跃。SAX API是一个基于事件的API,适用于处理数据流,即随着数据的流动而依次处理数据。SAX API
在其解析您的文档时发生一定事件的时候会通知您。在您对其响应时,您不作保存的数据将会 被抛弃。
下面是一个SAX解析XML的示例(有点长,因为详细注解了SAX事件处理的所有方法),SAX API中主要有四种处理事件的接口,它们分别是ContentHandler,DTDHandler,EntityResolver和ErrorHandler。下面的例子可能有点冗长,实际上只要继承DefaultHandler类 ,再覆盖一部分 处理事件的方法 同样可以达到这个示例的效果,但为了纵观全局,还是看看SAX API里面所有主要的事件解析方法吧。( 实际上DefaultHandler就是实现了上面的四个事件处理器接口,然后提供了每个抽象方法的默认实现。)
1,ContentHandler接口 :接收文档逻辑内容的通知 的处理器接口。
- importorg.xml.sax.Attributes;
- importorg.xml.sax.ContentHandler;
- importorg.xml.sax.Locator;
- importorg.xml.sax.SAXException;
- classMyContentHandlerimplementsContentHandler{
- StringBufferjsonStringBuffer;
- intfrontBlankCount=0;
- publicMyContentHandler(){
- jsonStringBuffer=newStringBuffer();
- }
- /*
- *接收字符数据的通知。
- *在DOM中ch[begin:end]相当于Text节点的节点值(nodeValue)
- */
- @Override
- publicvoidcharacters(char[]ch,intbegin,85); font-weight:bold">intlength)throwsSAXException{
- StringBufferbuffer=for(inti=begin;i<begin+length;i++){
- switch(ch[i]){
- case'\\':buffer.append("\\\\");break;
- case'\r':buffer.append("\\r");case'\n':buffer.append("\\n");case'\t':buffer.append("\\t");case'\"':buffer.append("\\\"");default:buffer.append(ch[i]);
- System.out.println(this.toBlankString(this.frontBlankCount)+
- ">>>characters("+length+"):"+buffer.toString());
- *接收文档的结尾的通知。
- voidendDocument()this.toBlankString(-- ">>>enddocument");
- *参数意义如下:
- *uri:元素的命名空间
- *localName:元素的本地名称(不带前缀)
- *qName:元素的限定名(带前缀)
- *
- voidendElement(Stringuri,StringlocalName,StringqName)
- ">>>endelement:"+qName+"("+uri+")");
- *结束前缀URI范围的映射。
- voidendPrefixMapping(Stringprefix) ">>>endprefix_mapping:"+prefix);
- *接收元素内容中可忽略的空白的通知。
- *ch:来自XML文档的字符
- *start:数组中的开始位置
- *length:从数组中读取的字符的个数
- voidignorableWhitespace(intlength)
- this.frontBlankCount)+">>>ignorablewhitespace("+length+"):"+buffer.toString());
- *接收处理指令的通知。
- *target:处理指令目标
- *data:处理指令数据,如果未提供,则为null。
- voidprocessingInstruction(Stringtarget,Stringdata)
- this.frontBlankCount)+">>>processinstruction:(target=\""
- +target+"\",data=\""+data+"\")");
- *接收用来查找SAX文档事件起源的对象。
- *locator:可以返回任何SAX文档事件位置的对象
- voidsetDocumentLocator(Locatorlocator){
- ">>>setdocument_locator:(lineNumber="+locator.getLineNumber()
- +",columnNumber="+locator.getColumnNumber()
- +locator.getSystemId()
- +locator.getPublicId()+")");
- *接收跳过的实体的通知。
- *name:所跳过的实体的名称。如果它是参数实体,则名称将以'%'开头,
- *如果它是外部DTD子集,则将是字符串"[dtd]"
- voidskippedEntity(Stringname) ">>>skipped_entity:"+name);
- *接收文档的开始的通知。
- voidstartDocument()this.frontBlankCount++)+
- ">>>startdocument");
- *接收元素开始的通知。
- *atts:元素的属性集合
- voidstartElement(Stringuri,StringqName,
- Attributesatts) ">>>startelement:"+qName+"("+uri+")");
- *开始前缀URI名称空间范围映射。
- *此事件的信息对于常规的命名空间处理并非必需:
- *当http://xml.org/sax/features/namespaces功能为true(默认)时,
- *SAXXML读取器将自动替换元素和属性名称的前缀。
- *prefix:前缀
- *uri:命名空间
- voidstartPrefixMapping(Stringprefix,Stringuri)
- ">>>startprefix_mapping:xmlns:"+prefix+"="
- +"\""+uri+"\"");
- privateStringtoBlankString(intcount){
- inti=0;i<count;i++)
- buffer.append("");
- returnbuffer.toString();
- }
2,DTDHandler接口 :接收与 DTD 相关的事件的通知的处理器接口。
3,EntityResolver接口 :是用于解析实体的基本接口。
4,ErrorHandler接口 :是错误处理程序的基本接口。
Test类的主方法打印解析books.xml时的事件信息。
books.xml文件的内容如下:
- <?xmlversion="1.0"encoding="GB2312"?>
- <bookscount="3"xmlns="http://test.org/books">
- <!--books'scomment-->
- bookid="1"name>ThinkinginJAVA</bookbookid="2">CoreJAVA2bookid="3">C++primerbooks>
控制台输出如下:
>>> set document_locator : (lineNumber = 1,columnNumber = 1,systemId = null,publicId = null)
>>> start document
Error (2,7) : Document is invalid: no grammar found.
Error (2,7) : Document root element "books",must match DOCTYPE root "null".
>>> start prefix_mapping : xmlns: = "http://test.org/books"
>>> start element : books()
>>> characters(2): \n\t
>>> characters(2): \n\t
>>> start element : book()
>>> characters(3): \n\t\t
>>> start element : name()
>>> characters(16): Thinking in JAVA
>>> end element : name()
>>> characters(2): \n\t
>>> end element : book()
>>> characters(2): \n\t
>>> start element : book()
>>> characters(10): Core JAVA2
>>> end element : name()
>>> characters(10): C++ primer
>>> end element : name()
>>> characters(1): \n
>>> end element : books() >>> end prefix_mapping : >>> end document