Sax解析xml_動(dòng)力節(jié)點(diǎn)Java學(xué)院整理
JAVA 解析 XML 通常有兩種方式,DOM 和 SAX。DOM 雖然是 W3C 的標(biāo)準(zhǔn),提供了標(biāo)準(zhǔn)的解析方式,但它的解析效率一直不盡如人意,因?yàn)槭褂肈OM解析XML時(shí),解析器讀入整個(gè)文檔并構(gòu)建一個(gè)駐留內(nèi)存的樹結(jié)構(gòu)(節(jié)點(diǎn)樹),然后您的代碼才可以使用 DOM 的標(biāo)準(zhǔn)接口來操作這個(gè)樹結(jié)構(gòu)。但大部分情況下我們只對(duì)文檔的部分內(nèi)容感興趣,根本就不用先解析整個(gè)文檔,并且從節(jié)點(diǎn)樹的根節(jié)點(diǎn)來索引一些我們需要的數(shù)據(jù)也是非常耗時(shí)的。
SAX是一種XML解析的替代方法。相比于文檔對(duì)象模型DOM,SAX 是讀取和操作 XML 數(shù)據(jù)的更快速、更輕量的方法。SAX 允許您在讀取文檔時(shí)處理它,從而不必等待整個(gè)文檔被存儲(chǔ)之后才采取操作。它不涉及 DOM 所必需的開銷和概念跳躍。 SAX API是一個(gè)基于事件的API ,適用于處理數(shù)據(jù)流,即隨著數(shù)據(jù)的流動(dòng)而依次處理數(shù)據(jù)。SAX API 在其解析您的文檔時(shí)發(fā)生一定事件的時(shí)候會(huì)通知您。在您對(duì)其響應(yīng)時(shí),您不作保存的數(shù)據(jù)將會(huì) 被拋棄。
下面是一個(gè)SAX解析XML的示例(有點(diǎn)長,因?yàn)樵敿?xì)注解了SAX事件處理的所有方法),SAX API中主要有四種處理事件的接口,它們分別是ContentHandler,DTDHandler, EntityResolver 和 ErrorHandler 。下面的例子可能有點(diǎn)冗長,實(shí)際上只要繼承DefaultHandler 類 ,再覆蓋一部分 處理事件的方法 同樣可以達(dá)到這個(gè)示例的效果,但為了縱觀全局,還是看看SAX API里面所有主要的事件解析方法吧。( 實(shí)際上DefaultHandler就是實(shí)現(xiàn)了上面的四個(gè)事件處理器接口,然后提供了每個(gè)抽象方法的默認(rèn)實(shí)現(xiàn)。)
1,ContentHandler 接口 :接收文檔邏輯內(nèi)容的通知 的處理器接口。
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(); } /* * 接收字符數(shù)據(jù)的通知。 * 在DOM中 ch[begin:end] 相當(dāng)于Text節(jié)點(diǎn)的節(jié)點(diǎn)值(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()); } /* * 接收文檔的結(jié)尾的通知。 */ @Override public void endDocument() throws SAXException { System.out.println(this.toBlankString(--this.frontBlankCount)+ ">>> end document"); } /* * 接收文檔的結(jié)尾的通知。 * 參數(shù)意義如下: * 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+")"); } /* * 結(jié)束前綴 URI 范圍的映射。 */ @Override public void endPrefixMapping(String prefix) throws SAXException { System.out.println(this.toBlankString(--this.frontBlankCount)+ ">>> end prefix_mapping : "+prefix); } /* * 接收元素內(nèi)容中可忽略的空白的通知。 * 參數(shù)意義如下: * ch : 來自 XML 文檔的字符 * start : 數(shù)組中的開始位置 * length : 從數(shù)組中讀取的字符的個(gè)數(shù) */ @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()); } /* * 接收處理指令的通知。 * 參數(shù)意義如下: * target : 處理指令目標(biāo) * data : 處理指令數(shù)據(jù),如果未提供,則為 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 文檔事件起源的對(duì)象。 * 參數(shù)意義如下: * locator : 可以返回任何 SAX 文檔事件位置的對(duì)象 */ @Override public void setDocumentLocator(Locator locator) { System.out.println(this.toBlankString(this.frontBlankCount)+ ">>> set document_locator : (lineNumber = "+locatorgetLineNumber() +",columnNumber = "+locatorgetColumnNumber() +",systemId = "+locatorgetSystemId() +",publicId = "+locatorgetPublicId()+")"); } /* * 接收跳過的實(shí)體的通知。 * 參數(shù)意義如下: * name : 所跳過的實(shí)體的名稱。如果它是參數(shù)實(shí)體,則名稱將以 '%' 開頭, * 如果它是外部 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 "); } /* * 接收元素開始的通知。 * 參數(shù)意義如下: * 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 名稱空間范圍映射。 * 此事件的信息對(duì)于常規(guī)的命名空間處理并非必需: * 當(dāng) http://xmlorg/sax/features/namespaces 功能為 true(默認(rèn))時(shí), * SAX XML 讀取器將自動(dòng)替換元素和屬性名稱的前綴。 * 參數(shù)意義如下: * 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 相關(guān)的事件的通知的處理器接口。
import org.xml.sax.DTDHandler; import org.xml.sax.SAXException; public class MyDTDHandler implements DTDHandler { /* * 接收注釋聲明事件的通知。 * 參數(shù)意義如下: * name - 注釋名稱。 * publicId - 注釋的公共標(biāo)識(shí)符,如果未提供,則為 null。 * systemId - 注釋的系統(tǒng)標(biāo)識(shí)符,如果未提供,則為 null。 */ @Override public void notationDecl(String name, String publicId, String systemId) throws SAXException { Systemoutprintln(">>> notation declare : (name = "+name +",systemId = "+publicId +",publicId = "+systemId+")"); } /* * 接收未解析的實(shí)體聲明事件的通知。 * 參數(shù)意義如下: * name - 未解析的實(shí)體的名稱。 * publicId - 實(shí)體的公共標(biāo)識(shí)符,如果未提供,則為 null。 * systemId - 實(shí)體的系統(tǒng)標(biāo)識(shí)符。 * notationName - 相關(guān)注釋的名稱。 */ @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 接口 :是用于解析實(shí)體的基本接口。
import java.io.IOException; import org.xml.sax.EntityResolver; import org.xml.sax.InputSource; import org.xml.sax.SAXException; public class MyEntityResolver implements EntityResolver { /* * 允許應(yīng)用程序解析外部實(shí)體。 * 解析器將在打開任何外部實(shí)體(頂級(jí)文檔實(shí)體除外)前調(diào)用此方法 * 參數(shù)意義如下: * publicId : 被引用的外部實(shí)體的公共標(biāo)識(shí)符,如果未提供,則為 null。 * systemId : 被引用的外部實(shí)體的系統(tǒng)標(biāo)識(shí)符。 * 返回: * 一個(gè)描述新輸入源的 InputSource 對(duì)象,或者返回 null, * 以請(qǐng)求解析器打開到系統(tǒng)標(biāo)識(shí)符的常規(guī) URI 連接。 */ @Override public InputSource resolveEntity(String publicId, String systemId) throws SAXException, IOException { return null; } }
4,ErrorHandler接口 :是錯(cuò)誤處理程序的基本接口。
import org.xml.sax.ErrorHandler; import org.xml.sax.SAXException; import org.xml.sax.SAXParseException; public class MyErrorHandler implements ErrorHandler { /* * 接收可恢復(fù)的錯(cuò)誤的通知 */ @Override public void error(SAXParseException e) throws SAXException { System.err.println("Error ("+e.getLineNumber()+"," +e.getColumnNumber()+") : "+e.getMessage()); } /* * 接收不可恢復(fù)的錯(cuò)誤的通知。 */ @Override public void fatalError(SAXParseException e) throws SAXException { System.err.println("FatalError ("+e.getLineNumber()+"," +e.getColumnNumber()+") : "+e.getMessage()); } /* * 接收不可恢復(fù)的錯(cuò)誤的通知。 */ @Override public void warning(SAXParseException e) throws SAXException { System.err.println("Warning ("+e.getLineNumber()+"," +e.getColumnNumber()+") : "+e.getMessage()); } }
Test 類的主方法打印解析books.xml時(shí)的事件信息。
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 { //創(chuàng)建處理文檔內(nèi)容相關(guān)事件的處理器 ContentHandler contentHandler = new MyContentHandler(); //創(chuàng)建處理錯(cuò)誤事件處理器 ErrorHandler errorHandler = new MyErrorHandler(); //創(chuàng)建處理DTD相關(guān)事件的處理器 DTDHandler dtdHandler = new MyDTDHandler(); //創(chuàng)建實(shí)體解析器 EntityResolver entityResolver = new MyEntityResolver(); //創(chuàng)建一個(gè)XML解析器(通過SAX方式讀取解析XML) XMLReader reader = XMLReaderFactory.createXMLReader(); /* * 設(shè)置解析器的相關(guān)特性 * http://xml.org/sax/features/validation = true 表示開啟驗(yàn)證特性 * 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); //設(shè)置XML解析器的處理文檔內(nèi)容相關(guān)事件的處理器 reader.setContentHandler(contentHandler); //設(shè)置XML解析器的處理錯(cuò)誤事件處理器 reader.setErrorHandler(errorHandler); //設(shè)置XML解析器的處理DTD相關(guān)事件的處理器 reader.setDTDHandler(dtdHandler); //設(shè)置XML解析器的實(shí)體解析器 reader.setEntityResolver(entityResolver); //解析books.xml文檔 reader.parse(new InputSource(new FileReader("books.xml"))); } }
books.xml 文件的內(nèi)容如下:
<?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>
控制臺(tái)輸出如下:
>>> 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
相關(guān)文章
Java實(shí)現(xiàn)STL中的全排列函數(shù)next_permutation()
在算法競(jìng)賽中,全排列問題是一個(gè)經(jīng)典且常見的題目,傳統(tǒng)的遞歸方法在處理較大的n時(shí)會(huì)遇到堆棧內(nèi)存限制的問題,本文介紹了一種避免遞歸,使用next_permutation函數(shù)實(shí)現(xiàn)全排列的方法,感興趣的朋友跟隨小編一起看看吧2024-09-09java 使用ElasticSearch完成百萬級(jí)數(shù)據(jù)查詢附近的人功能
本篇文章主要介紹了java 使用ElasticSearch完成百萬級(jí)數(shù)據(jù)查詢附近的人功能,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2018-01-01SpringBoot如何使用Fastjson解析Json數(shù)據(jù)
這篇文章主要介紹了SpringBoot如何使用Fastjson解析Json數(shù)據(jù),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-03-03Spring Boot使用Redisson實(shí)現(xiàn)滑動(dòng)窗口限流的項(xiàng)目實(shí)踐
滑動(dòng)窗口限流是一種流量控制策略,用于控制在一定時(shí)間內(nèi)的請(qǐng)求頻率,本文主要介紹了Spring Boot使用Redisson實(shí)現(xiàn)滑動(dòng)窗口限流的項(xiàng)目實(shí)踐,具有一定的參考價(jià)值,感興趣的可以了解一下2024-03-03Java將json對(duì)象轉(zhuǎn)換為map鍵值對(duì)案例詳解
這篇文章主要介紹了Java將json對(duì)象轉(zhuǎn)換為map鍵值對(duì)案例詳解,本篇文章通過簡(jiǎn)要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-09-09詳解Java 網(wǎng)絡(luò)IO編程總結(jié)(BIO、NIO、AIO均含完整實(shí)例代碼)
本篇文章主要介紹了Java 網(wǎng)絡(luò)IO編程總結(jié)(BIO、NIO、AIO均含完整實(shí)例代碼),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-12-12