Java使用SAX解析xml的示例
一、SAX解析xml簡介
SAX是Simple API for Xml的簡寫,主要功能是用于對xml文檔進(jìn)行解析。由于該方式采用的是事件驅(qū)動(callback回調(diào)機制)解析方式,所以有速度快、占內(nèi)存少的優(yōu)點,當(dāng)然這些優(yōu)點也僅限于xml的讀取操作,SAX是無法對讀取的XML元素進(jìn)行修改的。如果要修改節(jié)點元素則需要使用DOC方式進(jìn)行將xml文件讀取,它會將xml讀取成document樹結(jié)構(gòu)對象,這樣可用對節(jié)點元素進(jìn)行編輯操作;DOC方式的缺點也比較明顯:占內(nèi)存大、解析速度較慢。
所以僅用于讀取xml操作,使用SAX方式是比較好的方式。
二、SAX解析XML實例
創(chuàng)建一個解析的xml文件
<?xml version="1.0" encoding="utf-8"?> <persons> <user> <userId>1001</userId> <userName>張三</userName> </user> <user> <userId>1002</userId> <userName>李四</userName> </user> </persons>
創(chuàng)建一個XMLparseHandler用于自定義xml解析
public class Customhandler extends DefaultHandler2 { List<Map> list = new ArrayList<>(); Map map = null; String tag = ""; @Override public void startDocument() throws SAXException { System.out.println("開始解析xml"); } @Override public void startElement(String uri, String localName, String qName, Attributes attributes) throws SAXException { System.out.println("開始解析元素: <"+ qName + ">"); if(qName == "user"){ map = new HashMap(); } tag = qName; } @Override public void characters(char[] ch, int start, int length) throws SAXException { String text = new String(ch, start, length).trim(); if(text != null && !text.isEmpty() && tag!=null&& tag!=""){ map.put(tag, text); if(!map.containsKey(tag)){ } System.out.println("解析到元素值:"+ text); } } @Override public void endElement(String uri, String localName, String qName) throws SAXException { System.out.println("結(jié)束解析元素: <"+ qName + ">"); if(qName.equals("user")){ list.add(map); } tag = ""; } @Override public void endDocument() throws SAXException { System.out.println("結(jié)束解析xml"); } }
創(chuàng)建SAX解析對象解析xml
public static void main(String[] args) throws ParserConfigurationException, SAXException, IOException { //創(chuàng)建xml解析工廠 SAXParserFactory factory = SAXParserFactory.newInstance(); //創(chuàng)建xml解析對象 SAXParser parser = factory.newSAXParser(); File file = new File("test/custom/user.xml"); InputStream inputStream = new FileInputStream(file); Customhandler customhandler = new Customhandler(); //方式一 //parser.parse(inputStream, customhandler); //方式二 InputSource source = new InputSource(file.toURI().toURL().toString()); XMLReader xmlParser = parser.getXMLReader(); xmlParser.setContentHandler(customhandler); xmlParser.parse(source); List c = customhandler.list; inputStream.close(); } //打印結(jié)果為: 開始解析xml 開始解析元素: <persons> 開始解析元素: <user> 開始解析元素: <userId> 解析到元素值:1001 結(jié)束解析元素: <userId> 開始解析元素: <userName> 解析到元素值:張三 結(jié)束解析元素: <userName> 結(jié)束解析元素: <user> 開始解析元素: <user> 開始解析元素: <userId> 解析到元素值:1002 結(jié)束解析元素: <userId> 開始解析元素: <userName> 解析到元素值:李四 結(jié)束解析元素: <userName> 結(jié)束解析元素: <user> 結(jié)束解析元素: <persons> 結(jié)束解析xml
三、SAX的實際應(yīng)用
在tomcat源碼中,有一個Digester對象,這個Digester是tomcat啟動時,初始化各個容器(service、engine、Connetor)的執(zhí)行者,而Digester執(zhí)行容器初始化的依據(jù)是解析配置文件server.xml的內(nèi)容,根據(jù)xml的具體配置進(jìn)行來初始化容器。
下面是Digester的類的一些主要方法:
//org.apache.tomcat.util.digester.Digester#parse(org.xml.sax.InputSource) public class Digester extends DefaultHandler2 { //讀取解析xml public Object parse(InputSource input) throws IOException, SAXException { configure(); getXMLReader().parse(input); return root; } //對每個xml標(biāo)簽進(jìn)行解析,并執(zhí)行于之對應(yīng)的Rule規(guī)則列表 public void startElement(String namespaceURI, String localName, String qName, Attributes list) throws SAXException { boolean debug = log.isDebugEnabled(); // Parse system properties list = updateAttributes(list); // Save the body text accumulated for our surrounding element bodyTexts.push(bodyText); bodyText = new StringBuilder(); // the actual element name is either in localName or qName, depending // on whether the parser is namespace aware String name = localName; if ((name == null) || (name.length() < 1)) { name = qName; } // Compute the current matching rule StringBuilder sb = new StringBuilder(match); if (match.length() > 0) { sb.append('/'); } sb.append(name); //根據(jù)每次xml節(jié)點的名稱拼接成匹配url match = sb.toString(); // Fire "begin" events for all relevant rules(根據(jù)namespaceURI匹配獲取的Rule規(guī)則列表,有順序規(guī)則) List<Rule> rules = getRules().match(namespaceURI, match); matches.push(rules); if ((rules != null) && (rules.size() > 0)) { for (Rule value : rules) { try { Rule rule = value; if (debug) { log.debug(" Fire begin() for " + rule); } //依次執(zhí)行begin方法 rule.begin(namespaceURI, name, list); } catch (Exception e) { log.error("Begin event threw exception", e); throw createSAXException(e); } catch (Error e) { log.error("Begin event threw error", e); throw e; } } } else { if (debug) { log.debug(" No rules found matching '" + match + "'."); } } } }
與之對應(yīng)的server.xml片段如下:
<Service name="Catalina"> <Connector port="8081" protocol="HTTP/1.1" connectionTimeout="20000" redirectPort="8443" /> <Engine name="Catalina" defaultHost="localhost"> <Host name="localhost" appBase="webapps" unpackWARs="true" autoDeploy="true"> </Host> </Engine> </Service>
Digester讀取到上面這些xml標(biāo)簽后,就會從外向里進(jìn)行嵌套解析,將這些標(biāo)簽創(chuàng)建為與之對應(yīng)的java類實例,也就是tomcat的主體容器結(jié)構(gòu)。
以上就是Java使用SAX解析xml的示例的詳細(xì)內(nèi)容,更多關(guān)于Java使用SAX解析xml的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
詳解Java實現(xiàn)數(shù)據(jù)結(jié)構(gòu)之并查集
并查集這種數(shù)據(jù)結(jié)構(gòu),可能出現(xiàn)的頻率不是那么高,但是還會經(jīng)常性的見到,其理解學(xué)習(xí)起來非常容易,通過本文,一定能夠輕輕松松搞定并查集2021-06-06Java實現(xiàn)字符串轉(zhuǎn)換成可執(zhí)行代碼的方法
今天小編就為大家分享一篇Java實現(xiàn)字符串轉(zhuǎn)換成可執(zhí)行代碼的方法,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-07-07SpringCloud Feign Jackson自定義配置方式
這篇文章主要介紹了SpringCloud Feign Jackson自定義配置方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-03-03java 使用簡單的demo實例告訴你優(yōu)化算法的強大
本篇文章介紹了,在java中使用簡單的demo實例告訴你優(yōu)化算法的強大。需要的朋友參考下2013-05-05SpringBoot3整合SpringDoc實現(xiàn)在線接口文檔的詳細(xì)過程
這篇文章主要介紹了SpringBoot3整合SpringDoc實現(xiàn)在線接口文檔的詳細(xì)過程,本文通過示例代碼給大家介紹的非常詳細(xì),感興趣的朋友跟隨小編一起看看吧2024-06-06