`
步行者
  • 浏览: 170138 次
  • 性别: Icon_minigender_1
  • 来自: 武汉
社区版块
存档分类
最新评论

SAX解析XML 详解

    博客分类:
  • XML
阅读更多

    JAVA 解析 XML 通常有两种方式,DOMSAX。DOM 虽然是 W3C 的标准,提供了标准的解析方式,但它的解析效率一直不尽如人意,因为使用DOM解析XML时,解析器读入整个文档并构建一个驻留内存的树结构(节点树),然后您的代码才可以使用 DOM 的标准接口来操作这个树结构。但大部分情况下我们只对文档的部分内容感兴趣,根本就不用先解析整个文档,并且从节点树的根节点来索引一些我们需要的数据也是非常耗时的。 
    SAX是一种XML解析的替代方法。相比于文档对象模型DOM,SAX 是读取和操作 XML 数据的更快速、更轻量的方
法。SAX 允许您在读取文档时处理它,从而不必等待整个文档被存储之后才采取操作。它不涉及 DOM 所必需的开销和概念跳跃。 SAX API是一个基于事件的API ,适用于处理数据流,即随着数据的流动而依次处理数据。SAX API
在其解析您的文档时发生一定事件的时候会通知您。在您对其响应时,您不作保存的数据将会 被抛弃。
    下面是一个SAX解析XML的示例(有点长,因为详细注解了SAX事件处理的所有方法),SAX API中主要有四种处理事件的接口,它们分别是ContentHandlerDTDHandler EntityResolver ErrorHandler 。下面的例子可能有点冗长,实际上只要继承DefaultHandler 类 ,再覆盖一部分 处理事件的方法 同样可以达到这个示例的效果,但为了纵观全局,还是看看SAX API里面所有主要的事件解析方法吧。( 实际上DefaultHandler就是实现了上面的四个事件处理器接口,然后提供了每个抽象方法的默认实现。)

1,ContentHandler 接口 :接收文档逻辑内容的通知 的处理器接口。

import org.xml.sax.Attributes;
import org.xml.sax.ContentHandler;
import org.xml.sax.Locator;
import org.xml.sax.SAXException;

class MyContentHandler implements ContentHandler{
	StringBuffer jsonStringBuffer ;
	int frontBlankCount = 0;
	public MyContentHandler(){
		jsonStringBuffer = new StringBuffer();
	}
	/*
	 * 接收字符数据的通知。
	 * 在DOM中 ch[begin:end] 相当于Text节点的节点值(nodeValue)
	 */
	@Override
	public void characters(char[] ch, int begin, int length) throws SAXException {
		StringBuffer buffer = new StringBuffer();
		for(int i = begin ; i < begin+length ; i++){
			switch(ch[i]){
				case '\\':buffer.append("\\\\");break;
				case '\r':buffer.append("\\r");break;
				case '\n':buffer.append("\\n");break;
				case '\t':buffer.append("\\t");break;
				case '\"':buffer.append("\\\"");break;
				default : buffer.append(ch[i]);	
			}
		}
		System.out.println(this.toBlankString(this.frontBlankCount)+
				">>> characters("+length+"): "+buffer.toString());
	}

	
	/*
	 * 接收文档的结尾的通知。
	 */
	@Override
	public void endDocument() throws SAXException {
		System.out.println(this.toBlankString(--this.frontBlankCount)+
				">>> end document");
	}

	
	/*
	 * 接收文档的结尾的通知。
	 * 参数意义如下:
	 * 	  uri :元素的命名空间
	 *    localName :元素的本地名称(不带前缀)
	 *    qName :元素的限定名(带前缀)
	 * 
	 */
	@Override
	public void endElement(String uri,String localName,String qName)
			throws SAXException {
		System.out.println(this.toBlankString(--this.frontBlankCount)+
				">>> end element : "+qName+"("+uri+")");
	}

	/*
	 * 结束前缀 URI 范围的映射。
	 */
	@Override
	public void endPrefixMapping(String prefix) throws SAXException {
		System.out.println(this.toBlankString(--this.frontBlankCount)+
				">>> end prefix_mapping : "+prefix);
	}

	/*
	 * 接收元素内容中可忽略的空白的通知。
	 * 参数意义如下:
	 *     ch : 来自 XML 文档的字符
	 *     start : 数组中的开始位置
	 *     length : 从数组中读取的字符的个数
	 */
	@Override
	public void ignorableWhitespace(char[] ch, int begin, int length)
			throws SAXException {
		StringBuffer buffer = new StringBuffer();
		for(int i = begin ; i < begin+length ; i++){
			switch(ch[i]){
				case '\\':buffer.append("\\\\");break;
				case '\r':buffer.append("\\r");break;
				case '\n':buffer.append("\\n");break;
				case '\t':buffer.append("\\t");break;
				case '\"':buffer.append("\\\"");break;
				default : buffer.append(ch[i]);	
			}
		}
		System.out.println(this.toBlankString(this.frontBlankCount)+">>> ignorable whitespace("+length+"): "+buffer.toString());
	}
	
	/*
	 * 接收处理指令的通知。
	 * 参数意义如下:
	 *     target : 处理指令目标
	 *     data : 处理指令数据,如果未提供,则为 null。
	 */
	@Override
	public void processingInstruction(String target,String data)
			throws SAXException {
		System.out.println(this.toBlankString(this.frontBlankCount)+">>> process instruction : (target = \""
				+target+"\",data = \""+data+"\")");
	}

	/*
	 * 接收用来查找 SAX 文档事件起源的对象。
	 * 参数意义如下:
	 *     locator : 可以返回任何 SAX 文档事件位置的对象
	 */
	@Override
	public void setDocumentLocator(Locator locator) {
		System.out.println(this.toBlankString(this.frontBlankCount)+
				">>> set document_locator : (lineNumber = "+locator.getLineNumber()
				+",columnNumber = "+locator.getColumnNumber()
				+",systemId = "+locator.getSystemId()
				+",publicId = "+locator.getPublicId()+")");
		
	}

	/*
	 * 接收跳过的实体的通知。
	 * 参数意义如下: 
	 *     name : 所跳过的实体的名称。如果它是参数实体,则名称将以 '%' 开头,
	 *            如果它是外部 DTD 子集,则将是字符串 "[dtd]"
	 */
	@Override
	public void skippedEntity(String name) throws SAXException {
		System.out.println(this.toBlankString(this.frontBlankCount)+
				">>> skipped_entity : "+name);
	}

	/*
	 * 接收文档的开始的通知。
	 */
	@Override
	public void startDocument() throws SAXException {
		System.out.println(this.toBlankString(this.frontBlankCount++)+
				">>> start document ");
	}

	/*
	 * 接收元素开始的通知。
	 * 参数意义如下:
	 * 	  uri :元素的命名空间
	 *    localName :元素的本地名称(不带前缀)
	 *    qName :元素的限定名(带前缀)
	 *    atts :元素的属性集合
	 */
	@Override
	public void startElement(String uri, String localName, String qName, 
			Attributes atts) throws SAXException {
		System.out.println(this.toBlankString(this.frontBlankCount++)+
				">>> start element : "+qName+"("+uri+")");
	}
	
	/*
	 * 开始前缀 URI 名称空间范围映射。
	 * 此事件的信息对于常规的命名空间处理并非必需:
	 * 当 http://xml.org/sax/features/namespaces 功能为 true(默认)时,
	 * SAX XML 读取器将自动替换元素和属性名称的前缀。
	 * 参数意义如下:
	 *    prefix :前缀
	 * 	  uri :命名空间
	 */
	@Override
	public void startPrefixMapping(String prefix,String uri)
			throws SAXException {
		System.out.println(this.toBlankString(this.frontBlankCount++)+
				">>> start prefix_mapping : xmlns:"+prefix+" = "
				+"\""+uri+"\"");
		
	}
	
	private String toBlankString(int count){
		StringBuffer buffer = new StringBuffer();
		for(int i = 0;i<count;i++)
			buffer.append("    ");
		return buffer.toString();
	}
	
}

 

2,DTDHandler 接口 :接收与 DTD 相关的事件的通知的处理器接口。

import org.xml.sax.DTDHandler;
import org.xml.sax.SAXException;

public class MyDTDHandler implements DTDHandler {

	/*
	 * 接收注释声明事件的通知。
	 * 参数意义如下:
	 *     name - 注释名称。
	 *     publicId - 注释的公共标识符,如果未提供,则为 null。
	 *     systemId - 注释的系统标识符,如果未提供,则为 null。
	 */
	@Override
	public void notationDecl(String name, String publicId, String systemId)
			throws SAXException {
		System.out.println(">>> notation declare : (name = "+name
				+",systemId = "+publicId
				+",publicId = "+systemId+")");
	}

	/*
	 * 接收未解析的实体声明事件的通知。
	 * 参数意义如下:
	 *     name - 未解析的实体的名称。
	 *     publicId - 实体的公共标识符,如果未提供,则为 null。
	 *     systemId - 实体的系统标识符。
	 *     notationName - 相关注释的名称。
	 */
	@Override
	public void unparsedEntityDecl(String name,
            String publicId,
            String systemId,
            String notationName) throws SAXException {
		System.out.println(">>> unparsed entity declare : (name = "+name
				+",systemId = "+publicId
				+",publicId = "+systemId
				+",notationName = "+notationName+")");
	}

}

 


3,EntityResolver 接口 :是用于解析实体的基本接口。

import java.io.IOException;

import org.xml.sax.EntityResolver;
import org.xml.sax.InputSource;
import org.xml.sax.SAXException;

public class MyEntityResolver implements EntityResolver {

	/*
	 * 允许应用程序解析外部实体。
	 * 解析器将在打开任何外部实体(顶级文档实体除外)前调用此方法
	 * 参数意义如下:
	 *     publicId : 被引用的外部实体的公共标识符,如果未提供,则为 null。
	 *     systemId : 被引用的外部实体的系统标识符。
	 * 返回:
	 *     一个描述新输入源的 InputSource 对象,或者返回 null,
	 *     以请求解析器打开到系统标识符的常规 URI 连接。
	 */
	@Override
	public InputSource resolveEntity(String publicId, String systemId)
			throws SAXException, IOException {
		return null;
	}

}

 
4,ErrorHandler接口 :是错误处理程序的基本接口。

import org.xml.sax.ErrorHandler;
import org.xml.sax.SAXException;
import org.xml.sax.SAXParseException;

public class MyErrorHandler implements ErrorHandler {

	/*
	 * 接收可恢复的错误的通知
	 */
	@Override
	public void error(SAXParseException e) throws SAXException {
		System.err.println("Error ("+e.getLineNumber()+","
				+e.getColumnNumber()+") : "+e.getMessage());
	}
	
	/*
	 * 接收不可恢复的错误的通知。
	 */
	@Override
	public void fatalError(SAXParseException e) throws SAXException {
		System.err.println("FatalError ("+e.getLineNumber()+","
				+e.getColumnNumber()+") : "+e.getMessage());
	}

	/*
	 * 接收不可恢复的错误的通知。
	 */
	@Override
	public void warning(SAXParseException e) throws SAXException {
		System.err.println("Warning ("+e.getLineNumber()+","
				+e.getColumnNumber()+") : "+e.getMessage());
	}

}

 

Test 类的主方法打印解析books.xml时的事件信息。

import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.IOException;

import org.xml.sax.ContentHandler;
import org.xml.sax.DTDHandler;
import org.xml.sax.EntityResolver;
import org.xml.sax.ErrorHandler;
import org.xml.sax.InputSource;
import org.xml.sax.SAXException;
import org.xml.sax.XMLReader;
import org.xml.sax.helpers.XMLReaderFactory;


public class Test {

	public static void main(String[] args) throws SAXException, 
			FileNotFoundException, IOException {
		//创建处理文档内容相关事件的处理器
		ContentHandler contentHandler = new MyContentHandler();
		//创建处理错误事件处理器
		ErrorHandler errorHandler = new MyErrorHandler();
		//创建处理DTD相关事件的处理器
		DTDHandler dtdHandler = new MyDTDHandler();
		//创建实体解析器
		EntityResolver entityResolver = new MyEntityResolver();
		
		//创建一个XML解析器(通过SAX方式读取解析XML)
		XMLReader reader = XMLReaderFactory.createXMLReader(); 
		/*
		 * 设置解析器的相关特性
		 *     http://xml.org/sax/features/validation = true 表示开启验证特性
		 *     http://xml.org/sax/features/namespaces = true 表示开启命名空间特性
		 */
		reader.setFeature("http://xml.org/sax/features/validation",true);
		reader.setFeature("http://xml.org/sax/features/namespaces",true);
		//设置XML解析器的处理文档内容相关事件的处理器
		reader.setContentHandler(contentHandler);
		//设置XML解析器的处理错误事件处理器
		reader.setErrorHandler(errorHandler);
		//设置XML解析器的处理DTD相关事件的处理器
		reader.setDTDHandler(dtdHandler);
		//设置XML解析器的实体解析器
		reader.setEntityResolver(entityResolver);
		//解析books.xml文档
		reader.parse(new InputSource(new FileReader("books.xml")));
	}

}

 


books.xml 文件的内容如下:

 

<?xml version="1.0" encoding="GB2312"?>
<books  count="3" xmlns="http://test.org/books">
	<!--books's comment-->
	<book id="1">
		<name>Thinking in JAVA</name>
	</book>
	<book id="2">
		<name>Core JAVA2</name>
	</book>
	<book id="3">
		<name>C++ primer</name>
	</book>
</books>

 
控制台输出如下:

 

>>> 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(
http://test.org/books)
            >>> characters(2): \n\t
            >>> characters(2): \n\t
            >>> start element : book(
http://test.org/books)
                >>> characters(3): \n\t\t
                >>> start element : name(
http://test.org/books)
                    >>> characters(16): Thinking in JAVA
                >>> end element : name(
http://test.org/books)
                >>> characters(2): \n\t
            >>> end element : book(
http://test.org/books)
            >>> characters(2): \n\t
            >>> start element : book(
http://test.org/books)
                >>> characters(3): \n\t\t
                >>> start element : name(
http://test.org/books)
                    >>> characters(10): Core JAVA2
                >>> end element : name(
http://test.org/books)
                >>> characters(2): \n\t
            >>> end element : book(
http://test.org/books)
            >>> characters(2): \n\t
            >>> start element : book(
http://test.org/books)
                >>> characters(3): \n\t\t
                >>> start element : name(
http://test.org/books)
                    >>> characters(10): C++ primer
                >>> end element : name(
http://test.org/books)
                >>> characters(2): \n\t
            >>> end element : book(
http://test.org/books)
            >>> characters(1): \n
        >>> end element : books(
http://test.org/books)
    >>> end prefix_mapping :
>>> end document

分享到:
评论
4 楼 cmm13655612162 2014-07-04  
zhai4902176 写道
写的真不错,作为基础很薄弱的我理解的还是挺浅的。请问楼主 “DTD”指的是什么事件呢?

XML语法文件:DTD和XML Schema
3 楼 kisso143 2014-04-12  
写的很详细,不错,收藏一下
2 楼 zhai4902176 2014-01-14  
写的真不错,作为基础很薄弱的我理解的还是挺浅的。请问楼主 “DTD”指的是什么事件呢?
1 楼 zhuyz89 2012-04-21  
解析得很详细

相关推荐

    SAX解析XML

    **SAX解析XML详解** XML(eXtensible Markup Language)是一种用于标记数据的语言,广泛应用于数据交换、配置文件和Web服务等领域。SAX(Simple API for XML)是XML解析的一种方式,它采用事件驱动模型,以流式处理...

    android sax解析xml

    本篇将详细讲解如何在Android环境中使用SAX解析XML。 1. **SAX解析的基本原理** SAX解析器读取XML文档时,遇到每个元素、属性、文本等都会触发相应的事件,开发者需要注册事件处理器来处理这些事件。这种方式适合...

    使用SAX解析XML文件个人总结

    ### 使用SAX解析XML文件详解 #### 一、XML与SAX简介 XML(Extensible Markup Language,可扩展标记语言)是一种用于标记电子文件使其具有结构性的标记语言,旨在克服HTML的一些不足,使得数据能够被不同系统间共享...

    android学习笔记1--SAX解析XML

    ### Android学习笔记1—SAX解析XML #### SAX解析XML概览 在Android开发中,XML文件被广泛用于布局设计、资源定义等场景。而为了处理这些XML文件,开发者需要了解不同的XML解析技术,其中SAX(Simple API for XML)...

    sax解析xml实例

    ### SAX 解析 XML 实例详解 #### 一、SAX 解析器简介 SAX (Simple API for XML) 是一种基于事件驱动模型的 XML 解析技术。与 DOM (Document Object Model) 解析不同,SAX 不会将整个 XML 文档加载到内存中,而是...

    android SAX PULL 解析XML文件 代码 详解

    总之,Android提供了SAX和PULL两种解析XML的手段,开发者可以根据具体情况灵活选择。在内存有限或者需要高效处理大量XML数据的情况下,SAX可能是更好的选择;而在追求简洁代码和快速开发的场合,PULL解析则更为便捷...

    Android创建与解析XML(三)——详解Sax方式

    本文将深入讲解如何在Android中使用SAX解析XML以及如何通过SAX方式创建XML。 **1. SAX解析XML** SAX解析XML的基本流程是:创建SAXParserFactory,通过该工厂实例化SAXParser,然后设置事件处理器(ContentHandler...

    dom4j解析xml详解

    ### DOM4J解析XML详解 #### 一、DOM4J简介与特性 DOM4J是一个由dom4j.org开发的开源XML解析包,专为Java平台设计,它不仅支持DOM、SAX和JAXP标准,还巧妙地融入了Java集合框架,使其成为Java开发者在处理XML数据时...

    java解析XML详解

    Java作为一门流行的编程语言,提供了多种方式来解析XML文档,以便开发者能够有效地读取、操作和转换XML数据。本文将深入探讨Java解析XML的各种方法及其应用场景,旨在为读者提供全面而深入的理解。 ### Java解析XML...

    java解析XML详解.pdf

    Java解析XML技术详解: 在信息技术领域,XML(Extensible Markup Language,可扩展标记语言)是一种被广泛使用的标记语言,用于存储和传输数据。它与HTML相似,也是一种标记语言,但是HTML用于网页的格式化,而XML...

    Android编程使用sax解析xml数据的方法详解

    本篇文章将详细介绍如何在Android中使用SAX解析XML数据。 首先,我们需要一个XML文件作为解析的源数据。例如: ```xml &lt;?xml version="1.0" encoding="UTF-8"?&gt; &lt;name&gt;will &lt;age&gt;21 &lt;name&gt;will2 &lt;age&gt;...

    java_Dom4j解析XML详解

    Java DOM4J解析XML详解 XML(eXtensible Markup Language)是一种用于标记数据的语言,广泛应用于数据交换、配置文件和文档存储等领域。DOM4J是Java中一个强大的、轻量级的处理XML的库,它提供了丰富的API来读取、...

    解析xml详解及jar及源码

    本文将深入解析XML的基本概念、结构、解析方法,并介绍与之相关的Java库——Jar包及其源码。 XML的核心是其结构化特性,它通过元素(Element)、属性(Attribute)和文本内容来描述数据。每个XML文档都由一个根元素...

    详解android使用SAX解析XML文件

    本文将深入探讨如何使用SAX解析XML,这是一种轻量级且高效的方法,尤其适合内存和计算资源有限的环境,如Android设备。 SAX(Simple API for XML)解析器遵循事件驱动模式,与DOM(Document Object Model)解析器...

    xml解析与封装详解

    XML 解析与封装详解 XML(eXtensible Markup Language),即可扩展的标记性语言,是一种用于存储和传输数据的语言。XML 文件的所有标记都是可扩充的,不像 HTML 那样是固定的,而是由用户自行规定出来的。XML 的...

Global site tag (gtag.js) - Google Analytics