SAX解析XML 详解

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


目录结构图示: (注意jar包要下载导入)
SAX解析XML 详解_第1张图片


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

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

/**
 * @description SAX学习---ContentHandler接口  
 */
public class MyContentHandler implements ContentHandler{
    StringBuffer jsonStringBuffer ;  
    //为了让输出有层次感  定义每行输出前方的空格数
    int frontBlankCount = 0;  

    public MyContentHandler(){  
        jsonStringBuffer = new StringBuffer();  
    }  

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

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

    /**
     * @description 开始前缀 URI 名称空间范围映射。 此事件的信息对于常规的命名空间处理并非必需:
     *              当 http://xml.org/sax/features/namespaces 功能为 true(默认)时,SAX XML 读取器将自动替换元素和属性名称的前缀。   
     * @param prefix 前缀
     * @param uri 命名空间 
     * @throws SAXException
     */
    @Override
    public void startPrefixMapping(String prefix, String uri)
            throws SAXException {
        System.out.println(this.toBlankString(this.frontBlankCount++)+"startPrefixMapping()--- xmlns:"+prefix+" = "  +"\""+uri+"\"");  
    }

    /**
     * @description 接收文档的开始的通知。 
     * @param uri 元素的命名空间 
     * @param localName 元素的本地名称(不带前缀) 
     * @param qName 元素的限定名(带前缀) 
     * @param atts 元素的属性集合 
     * @throws SAXException
     */
    @Override
    public void startElement(String uri, String localName, String qName,
            Attributes atts) throws SAXException {
         System.out.println(this.toBlankString(this.frontBlankCount++) + "startElement()---"+qName+"      uri="+uri);   
    }

    /**
     * @description 接收字符数据的通知。 
     *              在DOM中 ch[start:end] 相当于Text节点的节点值(nodeValue) 
     * @param ch
     * @param start
     * @param length
     * @throws SAXException
     */
    @Override
    public void characters(char[] ch, int start, int length)
            throws SAXException {
        StringBuffer buffer = new StringBuffer();  
        for(int i = start ; i < start+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:"+ length +",str:"+buffer.toString());         
    }

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

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

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

    /**
     * @description 接收元素内容中可忽略的空白的通知。
     * @param ch 来自 XML 文档的字符 
     * @param start  数组中的开始位置 
     * @param length 从数组中读取的字符的个数 
     * @throws SAXException
     */
    @Override
    public void ignorableWhitespace(char[] ch, int start, int length)
            throws SAXException {
        StringBuffer buffer = new StringBuffer();  
        for(int i = start ; i < start+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("ignorableWhitespace()---" + this.toBlankString(this.frontBlankCount)+">>> ignorable whitespace("+length+"): "+buffer.toString());   
    }

    /**
     * @description  接收处理指令的通知。 
     * @param target 处理指令目标 
     * @param data 处理指令数据,如果未提供,则为 null。
     * @throws SAXException
     */
    @Override
    public void processingInstruction(String target, String data)
            throws SAXException {
        System.out.println("processingInstruction()---" + this.toBlankString(this.frontBlankCount)+">>> process instruction : (target = \"" +target+"\",data = \""+data+"\")");         
    }

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

    /**
     * @description 自写辅助工具 让输出有层次感觉 
     * @param count
     * @return
     */
    private String toBlankString(int count){  
        StringBuffer buffer = new StringBuffer();  
        for(int i = 0;i"    ");  
        return buffer.toString();  
    }
}

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

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

/**
 * @description SAX学习---DTDHandler接口,接收与 DTD相关的事件的通知的处理器接口
 */
public class MyDTDHandler implements DTDHandler{

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

    /**
     * @description 接收未解析的实体声明事件的通知。 
     * @param name 未解析的实体的名称。 
     * @param publicId 实体的公共标识符,如果未提供,则为 null。
     * @param systemId 实体的系统标识符。
     * @param notationName 相关注释的名称。 
     * @throws SAXException
     */
    @Override
    public void unparsedEntityDecl(String name, String publicId,
            String systemId, String notationName) throws SAXException {
        System.out.println("unparsedEntityDecl---(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;

/**
 * @description SAX学习---EntityResolver接口 
 */
public class MyEntityResolver implements EntityResolver{

    /**
     * @description 允许应用程序解析外部实体。 解析器将在打开任何外部实体(顶级文档实体除外)前调用此方法 
     * @param publicId 被引用的外部实体的公共标识符,如果未提供,则为 null。 
     * @param systemId 被引用的外部实体的系统标识符。 
     * @return 一个描述新输入源的 InputSource 对象,或者返回 null;以请求解析器打开到系统标识符的常规 URI 连接。 
     * @throws SAXException
     * @throws IOException 
     */
    @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;

/**
 * @description SAX学习---ErrorHandler接口,是错误处理程序的基本接口。
 */
public class MyErrorHandler implements ErrorHandler{

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

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

    /**
     * @description 接收不可恢复的错误的通知。  
     * @param exception
     * @throws SAXException
     * @author liuquan
     * @date  2015年12月11日
     */
    @Override
    public void fatalError(SAXParseException e) throws SAXException {
        System.err.println("FatalError ("+e.getLineNumber()+","+e.getColumnNumber()+") : "+e.getMessage());  
    }

}

5.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 FileNotFoundException, IOException, SAXException {
        //创建处理文档内容相关事件的处理器  
        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("book.xml")));  
    }
}

6.book.xml 文件的内容如下:


<books  count="3" xmlns="http://test.org/books">  
      
    <book id="1">  
        <name>Thinking in JAVAname>  
    book>  
    <book id="2">  
        <name>Core JAVA2name>  
    book>  
    <book id="3">  
        <name>C++ primername>  
    book>  
books>   

工程下载地址 (包括该文项目)

你可能感兴趣的:(SAX解析XML 详解)