加入收藏 | 设为首页 | 会员中心 | 我要投稿 李大同 (https://www.lidatong.com.cn/)- 科技、建站、经验、云计算、5G、大数据,站长网!
当前位置: 首页 > 百科 > 正文

SAX解析XML 详解

发布时间:2020-12-15 22:33:08 所属栏目:百科 来源:网络整理
导读:JAVA 解析 XML 通常有两种方式, DOM 和 SAX 。DOM 虽然是 W3C 的标准,提供了标准的解析方式,但它的解析效率一直不尽如人意,因为使用DOM解析XML时,解析器读入整个文档并构建一个驻留内存的树结构(节点树),然后您的代码才可以使用 DOM 的标准接口来操

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中主要有四种处理事件的接口,它们分别是ContentHandlerDTDHandlerEntityResolverErrorHandler。下面的例子可能有点冗长,实际上只要继承DefaultHandler类 ,再覆盖一部分 处理事件的方法 同样可以达到这个示例的效果,但为了纵观全局,还是看看SAX API里面所有主要的事件解析方法吧。( 实际上DefaultHandler就是实现了上面的四个事件处理器接口,然后提供了每个抽象方法的默认实现。)

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

Java代码
  1. importorg.xml.sax.Attributes;
  2. importorg.xml.sax.ContentHandler;
  3. importorg.xml.sax.Locator;
  4. importorg.xml.sax.SAXException;
  5. classMyContentHandlerimplementsContentHandler{
  6. StringBufferjsonStringBuffer;
  7. intfrontBlankCount=0;
  8. publicMyContentHandler(){
  9. jsonStringBuffer=newStringBuffer();
  10. }
  11. /*
  12. *接收字符数据的通知。
  13. *在DOM中ch[begin:end]相当于Text节点的节点值(nodeValue)
  14. */
  15. @Override
  16. publicvoidcharacters(char[]ch,intbegin,85); font-weight:bold">intlength)throwsSAXException{
  17. StringBufferbuffer=for(inti=begin;i<begin+length;i++){
  18. switch(ch[i]){
  19. case'':buffer.append("\");break;
  20. case'r':buffer.append("r");case'n':buffer.append("n");case't':buffer.append("t");case'"':buffer.append(""");default:buffer.append(ch[i]);
  21. System.out.println(this.toBlankString(this.frontBlankCount)+
  22. ">>>characters("+length+"):"+buffer.toString());
  23. *接收文档的结尾的通知。
  24. voidendDocument()this.toBlankString(-- ">>>enddocument");
  25. *参数意义如下:
  26. *uri:元素的命名空间
  27. *localName:元素的本地名称(不带前缀)
  28. *qName:元素的限定名(带前缀)
  29. *
  30. voidendElement(Stringuri,StringlocalName,StringqName)
  31. ">>>endelement:"+qName+"("+uri+")");
  32. *结束前缀URI范围的映射。
  33. voidendPrefixMapping(Stringprefix) ">>>endprefix_mapping:"+prefix);
  34. *接收元素内容中可忽略的空白的通知。
  35. *ch:来自XML文档的字符
  36. *start:数组中的开始位置
  37. *length:从数组中读取的字符的个数
  38. voidignorableWhitespace(intlength)
  39. this.frontBlankCount)+">>>ignorablewhitespace("+length+"):"+buffer.toString());
  40. *接收处理指令的通知。
  41. *target:处理指令目标
  42. *data:处理指令数据,如果未提供,则为null。
  43. voidprocessingInstruction(Stringtarget,Stringdata)
  44. this.frontBlankCount)+">>>processinstruction:(target=""
  45. +target+"",data=""+data+"")");
  46. *接收用来查找SAX文档事件起源的对象。
  47. *locator:可以返回任何SAX文档事件位置的对象
  48. voidsetDocumentLocator(Locatorlocator){
  49. ">>>setdocument_locator:(lineNumber="+locator.getLineNumber()
  50. +",columnNumber="+locator.getColumnNumber()
  51. +locator.getSystemId()
  52. +locator.getPublicId()+")");
  53. *接收跳过的实体的通知。
  54. *name:所跳过的实体的名称。如果它是参数实体,则名称将以'%'开头,
  55. *如果它是外部DTD子集,则将是字符串"[dtd]"
  56. voidskippedEntity(Stringname) ">>>skipped_entity:"+name);
  57. *接收文档的开始的通知。
  58. voidstartDocument()this.frontBlankCount++)+
  59. ">>>startdocument");
  60. *接收元素开始的通知。
  61. *atts:元素的属性集合
  62. voidstartElement(Stringuri,StringqName,
  63. Attributesatts) ">>>startelement:"+qName+"("+uri+")");
  64. *开始前缀URI名称空间范围映射。
  65. *此事件的信息对于常规的命名空间处理并非必需:
  66. *当http://xml.org/sax/features/namespaces功能为true(默认)时,
  67. *SAXXML读取器将自动替换元素和属性名称的前缀。
  68. *prefix:前缀
  69. *uri:命名空间
  70. voidstartPrefixMapping(Stringprefix,Stringuri)
  71. ">>>startprefix_mapping:xmlns:"+prefix+"="
  72. +"""+uri+""");
  73. privateStringtoBlankString(intcount){
  74. inti=0;i<count;i++)
  75. buffer.append("");
  76. returnbuffer.toString();
  77. }



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

importorg.xml.sax.DTDHandler;
  • classMyDTDHandlerimplementsDTDHandler{
  • *接收注释声明事件的通知。
  • *name-注释名称。
  • *publicId-注释的公共标识符,如果未提供,则为null。
  • *systemId-注释的系统标识符,如果未提供,则为null。
  • voidnotationDecl(Stringname,StringpublicId,StringsystemId)
  • System.out.println(">>>notationdeclare:(name="+name
  • +publicId
  • +systemId+")");
  • *接收未解析的实体声明事件的通知。
  • *name-未解析的实体的名称。
  • *publicId-实体的公共标识符,如果未提供,则为null。
  • *systemId-实体的系统标识符。
  • *notationName-相关注释的名称。
  • voidunparsedEntityDecl(Stringname,250)"> StringpublicId,250)"> StringsystemId,250)"> StringnotationName) System.out.println(">>>unparsedentitydeclare:(name="+name
  • +systemId
  • +notationName+")");

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

    importjava.io.IOException;
  • importorg.xml.sax.EntityResolver;
  • importorg.xml.sax.InputSource;
  • classMyEntityResolverimplementsEntityResolver{
  • *允许应用程序解析外部实体。
  • *解析器将在打开任何外部实体(顶级文档实体除外)前调用此方法
  • *publicId:被引用的外部实体的公共标识符,如果未提供,则为null。
  • *systemId:被引用的外部实体的系统标识符。
  • *返回:
  • *一个描述新输入源的InputSource对象,或者返回null,
  • *以请求解析器打开到系统标识符的常规URI连接。
  • publicInputSourceresolveEntity(StringpublicId,85); font-weight:bold">throwsSAXException,IOException{
  • returnnull;

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

    importorg.xml.sax.ErrorHandler;
  • importorg.xml.sax.SAXParseException;
  • classMyErrorHandlerimplementsErrorHandler{
  • *接收可恢复的错误的通知
  • voiderror(SAXParseExceptione) System.err.println("Error("+e.getLineNumber()+","
  • +e.getColumnNumber()+"):"+e.getMessage());
  • *接收不可恢复的错误的通知。
  • voidfatalError(SAXParseExceptione) System.err.println("FatalError("+e.getLineNumber()+",85); font-weight:bold">voidwarning(SAXParseExceptione) System.err.println("Warning("+e.getLineNumber()+",sans-serif; font-size:14px; line-height:25.200000762939453px">

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

    importjava.io.FileNotFoundException;
  • importjava.io.FileReader;
  • importorg.xml.sax.XMLReader;
  • importorg.xml.sax.helpers.XMLReaderFactory;
  • classTest{
  • staticvoidmain(String[]args) FileNotFoundException,0); padding:0px; margin:0px; width:auto; border:0px">//创建处理文档内容相关事件的处理器
  • ContentHandlercontentHandler=newMyContentHandler();
  • //创建处理错误事件处理器
  • ErrorHandlererrorHandler=newMyErrorHandler();
  • //创建处理DTD相关事件的处理器
  • DTDHandlerdtdHandler=newMyDTDHandler();
  • //创建实体解析器
  • EntityResolverentityResolver=newMyEntityResolver();
  • //创建一个XML解析器(通过SAX方式读取解析XML)
  • XMLReaderreader=XMLReaderFactory.createXMLReader();
  • *设置解析器的相关特性
  • *http://xml.org/sax/features/validation=true表示开启验证特性
  • *http://xml.org/sax/features/namespaces=true表示开启命名空间特性
  • reader.setFeature("http://xml.org/sax/features/validation",85); font-weight:bold">true);
  • reader.setFeature("http://xml.org/sax/features/namespaces",0); padding:0px; margin:0px; width:auto; border:0px">//设置XML解析器的处理文档内容相关事件的处理器
  • reader.setContentHandler(contentHandler);
  • //设置XML解析器的处理错误事件处理器
  • reader.setErrorHandler(errorHandler);
  • //设置XML解析器的处理DTD相关事件的处理器
  • reader.setDTDHandler(dtdHandler);
  • //设置XML解析器的实体解析器
  • reader.setEntityResolver(entityResolver);
  • //解析books.xml文档
  • reader.parse(newInputSource(newFileReader("books.xml")));

  • books.xml文件的内容如下:

    Xml代码
      <?xmlversion="1.0"encoding="GB2312"?>
    1. <bookscount="3"xmlns="http://test.org/books">
    2. <!--books'scomment-->
    3. 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): nt
    >>> characters(2): nt
    >>> start element : book(
    )
    >>> characters(3): ntt
    >>> start element : name(
    )
    >>> characters(16): Thinking in JAVA
    >>> end element : name(
    )
    >>> characters(2): nt
    >>> end element : book(
    )
    >>> characters(2): nt
    >>> 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

    (编辑:李大同)

    【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!

    • 推荐文章
        热点阅读