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 接口 :接收文檔邏輯內容的通知 的處理器接口。java
- 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();
- }
-
-
-
-
- @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");
- }
-
-
-
-
-
-
-
-
-
-
- @Override
- public void endElement(String uri,String localName,String qName)
- throws SAXException {
- System.out.println(this.toBlankString(--this.frontBlankCount)+
- ">>> end element : "+qName+"("+uri+")");
- }
-
-
-
-
- @Override
- public void endPrefixMapping(String prefix) throws SAXException {
- System.out.println(this.toBlankString(--this.frontBlankCount)+
- ">>> end prefix_mapping : "+prefix);
- }
-
-
-
-
-
-
-
-
- @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());
- }
-
-
-
-
-
-
-
- @Override
- public void processingInstruction(String target,String data)
- throws SAXException {
- System.out.println(this.toBlankString(this.frontBlankCount)+">>> process instruction : (target = \""
- +target+"\",data = \""+data+"\")");
- }
-
-
-
-
-
-
- @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()+")");
-
- }
-
-
-
-
-
-
-
- @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 ");
- }
-
-
-
-
-
-
-
-
-
- @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+")");
- }
-
-
-
-
-
-
-
-
-
-
- @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();
- }
-
- }
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 相關的事件的通知的處理器接口。node
- import org.xml.sax.DTDHandler;
- import org.xml.sax.SAXException;
-
- public class MyDTDHandler implements DTDHandler {
-
-
-
-
-
-
-
-
- @Override
- public void notationDecl(String name, String publicId, String systemId)
- throws SAXException {
- System.out.println(">>> notation declare : (name = "+name
- +",systemId = "+publicId
- +",publicId = "+systemId+")");
- }
-
-
-
-
-
-
-
-
-
- @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+")");
- }
-
- }
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 接口 :是用於解析實體的基本接口。json
- import java.io.IOException;
-
- import org.xml.sax.EntityResolver;
- import org.xml.sax.InputSource;
- import org.xml.sax.SAXException;
-
- public class MyEntityResolver implements EntityResolver {
-
-
-
-
-
-
-
-
-
-
-
- @Override
- public InputSource resolveEntity(String publicId, String systemId)
- throws SAXException, IOException {
- return null;
- }
-
- }
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());
- }
-
- }
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時的事件信息。app
- 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();
-
- DTDHandler dtdHandler = new MyDTDHandler();
-
- EntityResolver entityResolver = new MyEntityResolver();
-
-
- XMLReader reader = XMLReaderFactory.createXMLReader();
-
-
-
-
-
- reader.setFeature("http://xml.org/sax/features/validation",true);
- reader.setFeature("http://xml.org/sax/features/namespaces",true);
-
- reader.setContentHandler(contentHandler);
-
- reader.setErrorHandler(errorHandler);
-
- reader.setDTDHandler(dtdHandler);
-
- reader.setEntityResolver(entityResolver);
-
- reader.parse(new InputSource(new FileReader("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 文件的內容以下: ide
- <?xml version="1.0" encoding="GB2312"?>
- <books count="3" xmlns="http://test.org/books">
-
- <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>
<?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>
控制檯輸出以下: this
>>> 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