Java使用SAX解析xml的示例
一、SAX解析xml簡介
SAX是Simple API for Xml的簡寫,主要功能是用于對xml文檔進(jìn)行解析。由于該方式采用的是事件驅(qū)動(callback回調(diào)機(jī)制)解析方式,所以有速度快、占內(nèi)存少的優(yōu)點(diǎn),當(dāng)然這些優(yōu)點(diǎn)也僅限于xml的讀取操作,SAX是無法對讀取的XML元素進(jìn)行修改的。如果要修改節(jié)點(diǎn)元素則需要使用DOC方式進(jìn)行將xml文件讀取,它會將xml讀取成document樹結(jié)構(gòu)對象,這樣可用對節(jié)點(diǎn)元素進(jìn)行編輯操作;DOC方式的缺點(diǎn)也比較明顯:占內(nèi)存大、解析速度較慢。
所以僅用于讀取xml操作,使用SAX方式是比較好的方式。
二、SAX解析XML實(shí)例
創(chuàng)建一個(gè)解析的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)建一個(gè)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的實(shí)際應(yīng)用
在tomcat源碼中,有一個(gè)Digester對象,這個(gè)Digester是tomcat啟動時(shí),初始化各個(gè)容器(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;
}
//對每個(gè)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é)點(diǎn)的名稱拼接成匹配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類實(shí)例,也就是tomcat的主體容器結(jié)構(gòu)。
以上就是Java使用SAX解析xml的示例的詳細(xì)內(nèi)容,更多關(guān)于Java使用SAX解析xml的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
詳解Java實(shí)現(xiàn)數(shù)據(jù)結(jié)構(gòu)之并查集
并查集這種數(shù)據(jù)結(jié)構(gòu),可能出現(xiàn)的頻率不是那么高,但是還會經(jīng)常性的見到,其理解學(xué)習(xí)起來非常容易,通過本文,一定能夠輕輕松松搞定并查集2021-06-06
Java實(shí)現(xiàn)字符串轉(zhuǎn)換成可執(zhí)行代碼的方法
今天小編就為大家分享一篇Java實(shí)現(xiàn)字符串轉(zhuǎn)換成可執(zhí)行代碼的方法,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-07-07
運(yùn)行jar程序時(shí)添加vm參數(shù)的方法
下面小編就為大家?guī)硪黄\(yùn)行jar程序時(shí)添加vm參數(shù)的方法。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2017-02-02
SpringCloud Feign Jackson自定義配置方式
這篇文章主要介紹了SpringCloud Feign Jackson自定義配置方式,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-03-03
java 使用簡單的demo實(shí)例告訴你優(yōu)化算法的強(qiáng)大
本篇文章介紹了,在java中使用簡單的demo實(shí)例告訴你優(yōu)化算法的強(qiáng)大。需要的朋友參考下2013-05-05
SpringBoot3整合SpringDoc實(shí)現(xiàn)在線接口文檔的詳細(xì)過程
這篇文章主要介紹了SpringBoot3整合SpringDoc實(shí)現(xiàn)在線接口文檔的詳細(xì)過程,本文通過示例代碼給大家介紹的非常詳細(xì),感興趣的朋友跟隨小編一起看看吧2024-06-06

