Java使用SAX解析xml的示例
一、SAX解析xml簡(jiǎn)介
SAX是Simple API for Xml的簡(jiǎn)寫,主要功能是用于對(duì)xml文檔進(jìn)行解析。由于該方式采用的是事件驅(qū)動(dòng)(callback回調(diào)機(jī)制)解析方式,所以有速度快、占內(nèi)存少的優(yōu)點(diǎn),當(dāng)然這些優(yōu)點(diǎn)也僅限于xml的讀取操作,SAX是無(wú)法對(duì)讀取的XML元素進(jìn)行修改的。如果要修改節(jié)點(diǎn)元素則需要使用DOC方式進(jìn)行將xml文件讀取,它會(huì)將xml讀取成document樹結(jié)構(gòu)對(duì)象,這樣可用對(duì)節(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解析對(duì)象解析xml
public static void main(String[] args) throws ParserConfigurationException, SAXException, IOException {
//創(chuàng)建xml解析工廠
SAXParserFactory factory = SAXParserFactory.newInstance();
//創(chuàng)建xml解析對(duì)象
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對(duì)象,這個(gè)Digester是tomcat啟動(dòng)時(shí),初始化各個(gè)容器(service、engine、Connetor)的執(zhí)行者,而Digester執(zhí)行容器初始化的依據(jù)是解析配置文件server.xml的內(nèi)容,根據(jù)xml的具體配置進(jìn)行來(lái)初始化容器。
下面是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;
}
//對(duì)每個(gè)xml標(biāo)簽進(jìn)行解析,并執(zhí)行于之對(duì)應(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 + "'.");
}
}
}
}
與之對(duì)應(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)簽后,就會(huì)從外向里進(jìn)行嵌套解析,將這些標(biāo)簽創(chuàng)建為與之對(duì)應(yīng)的java類實(shí)例,也就是tomcat的主體容器結(jié)構(gòu)。
以上就是Java使用SAX解析xml的示例的詳細(xì)內(nèi)容,更多關(guān)于Java使用SAX解析xml的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Java正則驗(yàn)證正整數(shù)的方法分析【測(cè)試可用】
這篇文章主要介紹了Java正則驗(yàn)證正整數(shù)的方法,結(jié)合實(shí)例形式對(duì)比分析了java針對(duì)正整數(shù)的驗(yàn)證方法及相關(guān)注意事項(xiàng),需要的朋友可以參考下2017-08-08
mysql數(shù)據(jù)庫(kù)忘記密碼時(shí)如何修改
本文主要介紹了mysql數(shù)據(jù)庫(kù)忘記密碼時(shí)如何修改的步驟方法,具有很好的參考價(jià)值,下面跟著小編一起來(lái)看下吧2017-02-02
Java 反射獲取類詳細(xì)信息的常用方法總結(jié)
Java 反射獲取類詳細(xì)信息的常用方法總結(jié),需要的朋友可以參考一下2013-03-03
Java使用JDBC向MySQL數(shù)據(jù)庫(kù)批次插入10W條數(shù)據(jù)(測(cè)試效率)
使用JDBC連接MySQL數(shù)據(jù)庫(kù)進(jìn)行數(shù)據(jù)插入的時(shí)候,特別是大批量數(shù)據(jù)連續(xù)插入(100000),如何提高效率呢?今天小編通過(guò)本教程給大家介紹下2016-12-12
idea插件之如何使用JarEditor編輯Java JAR文件
JarEditor是一款用于在IntelliJIDEA中直接編輯JAR文件的插件,支持反編譯查看和編輯.class文件,并提供即時(shí)編譯與保存功能,通過(guò)JarEditor,用戶可以在IDE內(nèi)一站式完成JAR文件的編輯、管理和打包操作,提高開發(fā)效率,但在生產(chǎn)環(huán)境中使用前,請(qǐng)確保備份并測(cè)試修改2025-01-01

