Sax解析xml_动力节点Java学院整理

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 接口 :接收文档逻辑内容的通知 的处理器接口。

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+"): "+buffertoString());
 } 

 /*
  * 接收文档的结尾的通知。
  */
 @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 '\\':bufferappend("\\\\");break;
    case '\r':bufferappend("\\r");break;
    case '\n':bufferappend("\\n");break;
    case '\t':bufferappend("\\t");break;
    case '\"':bufferappend("\\\"");break;
    default : bufferappend(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 = "+locatorgetLineNumber()
    +",columnNumber = "+locatorgetColumnNumber()
    +",systemId = "+locatorgetSystemId()
    +",publicId = "+locatorgetPublicId()+")"); 

 } 

 /*
  * 接收跳过的实体的通知。
  * 参数意义如下:
  *  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://xmlorg/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 {
  Systemoutprintln(">>> 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 {
  Systemoutprintln(">>> 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://testorg/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

(0)

相关推荐

  • java中使用sax解析xml的解决方法

    在java中,原生解析xml文档的方式有两种,分别是:Dom解析和Sax解析 Dom解析功能强大,可增删改查,操作时会将xml文档以文档对象的方式读取到内存中,因此适用于小文档 Sax解析是从头到尾逐行逐个元素读取内容,修改较为不便,但适用于只读的大文档 本文主要讲解Sax解析,其余放在后面 Sax采用事件驱动的方式解析文档.简单点说,如同在电影院看电影一样,从头到尾看一遍就完了,不能回退(Dom可来来回回读取) 在看电影的过程中,每遇到一个情节,一段泪水,一次擦肩,你都会调动大脑和神经去接收或

  • Java下3中XML解析 DOM方式、SAX方式和StAX方式

    先简单说下前三种方式: DOM方式:个人理解类似.net的XmlDocument,解析的时候效率不高,占用内存,不适合大XML的解析:SAX方式:基于事件的解析,当解析到xml的某个部分的时候,会触发特定事件,可以在自定义的解析类中定义当事件触发时要做得事情:个人感觉一种很另类的方式,不知道.Net体系下是否有没有类似的方式?StAX方式:个人理解类似.net的XmlReader方式,效率高,占用内存少,适用大XML的解析:不过SAX方式之前也用过,本文主要介绍JAXB,这里只贴下主要代码: 复

  • Java中使用DOM和SAX解析XML文件的方法示例

    dom4j介绍 dom4j的项目地址:http://sourceforge.net/projects/dom4j/?source=directory dom4j是一个简单的开源库,用于处理XML. XPath和XSLT,它基于Java平台,使用Java的集合框架,全面集成了DOM,SAX和JAXP. dom4j的使用 下载了dom4j项目之后,解压缩,将其jar包(我的当前版本叫做dom4j-1.6.1.jar)加入class path下面. (Properties->Java Build Pa

  • java解析xml之sax解析xml示例分享

    复制代码 代码如下: package com.test; import java.io.File;import java.io.FileInputStream;import java.util.ArrayList;import java.util.List; import javax.xml.parsers.SAXParser;import javax.xml.parsers.SAXParserFactory; import org.xml.sax.Attributes;import org.x

  • Sax解析xml_动力节点Java学院整理

    JAVA 解析 XML 通常有两种方式,DOM 和 SAX.DOM 虽然是 W3C 的标准,提供了标准的解析方式,但它的解析效率一直不尽如人意,因为使用DOM解析XML时,解析器读入整个文档并构建一个驻留内存的树结构(节点树),然后您的代码才可以使用 DOM 的标准接口来操作这个树结构.但大部分情况下我们只对文档的部分内容感兴趣,根本就不用先解析整个文档,并且从节点树的根节点来索引一些我们需要的数据也是非常耗时的. SAX是一种XML解析的替代方法.相比于文档对象模型DOM,SAX 是读取和操作

  • Java中使用jaxp进行sax解析_动力节点Java学院整理

    SAX解析XML文件采用事件驱动的方式进行,也就是说,SAX是逐行扫描文件,遇到符合条件的设定条件后就会触发特定的事件,回调你写好的事件处理程序.使用SAX的优势在于其解析速度较快,相对于DOM而言占用内存较少.而且SAX在解析文件的过程中得到自己需要的信息后可以随时终止解析,并不一定要等文件全部解析完毕.凡事有利必有弊,其劣势在于SAX采用的是流式处理方式,当遇到某个标签的时候,它并不会记录下以前所遇到的标签,也就是说,在处理某个标签的时候,比如在startElement方法中,所能够得到的信

  • Java中JDom解析XML_动力节点Java学院整理

    一.前言 JDOM是Breet Mclaughlin和Jason Hunter两大Java高手的创作成果,2000年初,JDOM作为一个开放源代码项目正式开始研发.JDOM是一种解析XML的Java工具包. DOM适合于当今流行的各种语言,包括Java,JavaScripte,VB,VBScript,Perl,C,C++等.它了为HTML和XML文档提供了一个可应用于不同平台的编程接口.W3C DOM的最新信息可从http://www.w3.org/TR2001/WD-DOM-Lever-3-C

  • Dom4j解析XML_动力节点Java学院整理

    dom4j是一个Java的XML API,类似于jdom,用来读写XML文件的.dom4j是一个非常非常优秀的Java XML API,具有性能优异.功能强大和极端易用使用的特点,同时它也是一个开放源代码的软件,可以在SourceForge上找到它. 对主流的java XML API进行的性能.功能和易用性的评测,dom4j无论在那个方面都是非常出色的.如今你可以看到越来越多的Java软件都在使用dom4j来读写XML,例如hibernate,包括sun公司自己的JAXM也用了Dom4j. 使用

  • Java异常继承结构解析_动力节点Java学院整理

    Java异常类层次结构图: 异常的英文单词是exception,字面翻译就是"意外.例外"的意思,也就是非正常情况.事实上,异常本质上是程序上的错误,包括程序逻辑错误和系统错误.比如使用空的引用.数组下标越界.内存溢出错误等,这些都是意外的情况,背离我们程序本身的意图.错误在我们编写程序的过程中会经常发生,包括编译期间和运行期间的错误,在编译期间出现的错误有编译器帮助我们一起修正,然而运行期间的错误便不是编译器力所能及了,并且运行期间的错误往往是难以预料的.假若程序在运行期间出现了错误

  • 使用jaxp进行dom解析_动力节点Java学院整理

    1.javax.xml.parsers 包中的DocumentBuilderFactory用于创建DOM模式的解析器对象 , DocumentBuilderFactory是一个抽象工厂类,它不能直接实例化,但该类提供了一个newInstance方法 ,这个方法会根据本地平台默认安装的解析器,自动创建一个工厂的对象并返回 2.调用 DocumentBuilderFactory.newInstance() 方法得到创建 DOM 解析器的工厂. 3.调用工厂对象的 newDocumentBuilder

  • Nginx简介_动力节点Java学院整理

    1.什么是Nginx Nginx来自俄罗斯的Igor Sysoev在为Rambler Media(http://www.rambler.ru/)工作期间,使用C语言开发了Nginx.Nginx作为Web服务器,一直为俄罗斯著名的门户网站Rambler Media提供着出色.稳定的服务. Igor Sysoev将Nginx的代码开源,并且赋予其最自由的2-clause BSD-like license许可证.由于Nginx使用基于事件驱动的架构能够并发处理百万级别的TCP连接,高度模块化的设计和自

  • 十大常见Java String问题_动力节点Java学院整理

    本文介绍Java中关于String最常见的10个问题: 1. 字符串比较,使用 "==" 还是 equals() ? 简单来说, "==" 判断两个引用的是不是同一个内存地址(同一个物理对象). 而 equals 判断两个字符串的值是否相等. 除非你想判断两个string引用是否同一个对象,否则应该总是使用 equals()方法. 如果你了解 字符串的驻留 ( String Interning ) 则会更好地理解这个问题 2. 对于敏感信息,为何使用char[]要比

  • Java JVM原理与调优_动力节点Java学院整理

    JVM是一种用于计算设备的规范,它是一个虚构出来的计算机,是通过在实际的计算机上仿真模拟各种计算机功能来实现的.Java虚拟机包括一套字节码指令集.一组寄存器.一个栈.一个垃圾回收堆和一个存储方法域. JVM屏蔽了与具体操作系统平台相关的信息,使Java程序只需生成在Java虚拟机上运行的目标代码(字节码),就可以在多种平台上不加修改地运行.是运行Java应用最底层部分. JDK(Java Development kit) 整个Java的核心,包括了Java运行环境(Java Runtime E

  • Java中的HashSet详解和使用示例_动力节点Java学院整理

    第1部分 HashSet介绍 HashSet 简介 HashSet 是一个没有重复元素的集合. 它是由HashMap实现的,不保证元素的顺序,而且HashSet允许使用 null 元素. HashSet是非同步的.如果多个线程同时访问一个哈希 set,而其中至少一个线程修改了该 set,那么它必须 保持外部同步.这通常是通过对自然封装该 set 的对象执行同步操作来完成的.如果不存在这样的对象,则应该使用 Collections.synchronizedSet 方法来"包装" set.

随机推荐