[Java开发之路](11)SAX解析XML文档

本文涉及的产品
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
全局流量管理 GTM,标准版 1个月
云解析 DNS,旗舰版 1个月
简介:


1. 简介

Dom解析功能强大,可增删改查,操作时会将XML文档读到内存,因此适用于小文档;
SAX解析是从头到尾逐行逐个元素解析,修改较为不便,但适用于只读的大文档;

SAX采用事件驱动的方式解析XML。套用网友的解释:如同在电影院看电影一样,从头到尾看一遍,不能回退(Dom可来来回回读取), 在看电影的过程中,每遇到一个情节,都会调用大脑去接收处理这些信息。SAX也是相同的原理,每遇到一个元素节点,都会调用相应的方法来处理。 在SAX的解析过程中,读取到文档开头、文档结尾,元素的开头和元素结尾都会调用相应方法,我们可以在这些方法中进行相应事件处理。

对应方法:
 
   
public void startDocument() throws SAXException {
}
public void endDocument() throws SAXException {
}
public void startElement(String uri, String localName, String qName, Attributes attributes) throws SAXException {
}
public void endElement(String uri, String localName, String qName) throws SAXException {
}

我们还需一个方法来处理元素节点中间的文本节点(我们常误以为元素节点的文本值)
 
  
public void characters(char[] ch, int start, int length) throws SAXException {
}

2. 解析

解析步骤:

(1)通过SAXParserFactory的静态方法newInstance()方法获取SAXParserFactory实例对象factory
 
  
SAXParserFactory factory = SAXParserFactory.newInstance();
(2)通过SAXParserFactory实例的newSAXParser()方法返回SAXParser实例parser
 
  
SAXParser parser = factory.newSAXParser();
(3)创建一个类继承DefaultHandler,重写其中的一些方法进行业务处理
 
  
package com.qunar.handler;
import org.xml.sax.Attributes;
import org.xml.sax.SAXException;
import org.xml.sax.helpers.DefaultHandler;
public class SAXParserHandler extends DefaultHandler{
// 用来标示解析开始
@Override
public void startDocument() throws SAXException {
}
// 用来标示解析结束
@Override
public void endDocument() throws SAXException {
}
// 用来遍历XML文件的开始标签
@Override
public void startElement(String uri, String localName, String qName, Attributes attributes) throws SAXException {
super.startElement(uri, localName, qName, attributes);
}
// 用来遍历XML文件的结束标签
@Override
public void endElement(String uri, String localName, String qName) throws SAXException {
super.endElement(uri, localName, qName);
}
@Override
public void characters(char[] ch, int start, int length) throws SAXException {
super.characters(ch, start, length);
}
}
(4)创建Handler类对象实例
 
  
// 定义SAXParserHandler对象
SAXParserHandler handler = new SAXParserHandler();
(5)解析XML文档
 
  
saxParser.parse(path, handler);


以下代码均使用本XML文档:
 
  
<?xml version="1.0" encoding="utf-8"?><bookstore>
<book category="Java">
<title lang="chi">Java多线程编程核心技术</title>
<author>高洪岩</author>
<year>2015</year>
<price>69.00</price>
</book>
<book category="C++">
<title lang="en">Effective C++: 55 Specific Ways to Improve Your Programs and Designs</title>
<author>Scott Meyers</author>
<year>2006</year>
<price>58.00</price>
</book>
<book category="Web">
<title lang="en">Learning XML</title>
<author>Erik T. Ray</author>
<year>2016</year>
<price>39.95</price>
</book>
</bookstore>

3. 具体实例:
 
  
package com.qunar.handler;
 
import org.xml.sax.Attributes;
import org.xml.sax.SAXException;
import org.xml.sax.helpers.DefaultHandler;
 
 
public class SAXParserHandler extends DefaultHandler{
private int bookIndex = 0;
// 用来标示解析开始
@Override
public void startDocument() throws SAXException {
System.out.println("SAX解析开始...");
}
// 用来标示解析结束
@Override
public void endDocument() throws SAXException {
System.out.println("SAX解析结束...");
}
// 用来遍历XML文件的开始标签
@Override
public void startElement(String uri, String localName, String qName, Attributes attributes) throws SAXException {
// 调用DefaultHandler类的startElement方法
super.startElement(uri, localName, qName, attributes);
// 开始解析book元素节点
if(qName.equals("book")){
++ bookIndex;
System.out.println("开始解析第" + bookIndex + "本书...");
// 已知book元素节点下的属性名称,根据属性名称获取属性值
/*String value = attributes.getValue("category");
System.out.println("value->"+value);*/
// 不知道book元素节点下的属性名称以及个数
int size = attributes.getLength();
for(int i = 0;i < size;++i){
System.out.println(attributes.getQName(i) + ":" + attributes.getValue(i));
}//for
}//if
else if(!qName.equals("bookstore")){
System.out.print(qName + ":");
}//else
}
// 用来遍历XML文件的结束标签
@Override
public void endElement(String uri, String localName, String qName) throws SAXException {
super.endElement(uri, localName, qName);
// 判断一本书是否解析完
if(qName.equals("book")){
System.out.println("结束解析第" + bookIndex + "本书...");
}//if
}
@Override
public void characters(char[] ch, int start, int length) throws SAXException {
super.characters(ch, start, length);
String text = new String(ch, start, length);
if(!text.trim().equals("")){
System.out.println(text);
}//if
}
}

 
  
package com.qunar.xml;
 
import java.io.IOException;
 
import javax.xml.parsers.ParserConfigurationException;
import javax.xml.parsers.SAXParser;
import javax.xml.parsers.SAXParserFactory;
 
import org.xml.sax.SAXException;
 
import com.qunar.handler.SAXParserHandler;
 
 
/**
* SAX方式解析XML文档
* @author sjf0115
*
*/
public class SAXXMLCode {
 
public static void main(String[] args) {
String path = "D:\\bookstore.xml";
try {
// 通过SAXParserFactory的静态方法newInstance()方法获取SAXParserFactory实例对象factory
SAXParserFactory factory = SAXParserFactory.newInstance();
// 通过SAXParserFactory实例的newSAXParser()方法返回SAXParser实例parser
SAXParser saxParser = factory.newSAXParser();
// 定义SAXParserHandler对象
SAXParserHandler handler = new SAXParserHandler();
// 解析XML文档
saxParser.parse(path, handler);
} catch (ParserConfigurationException e) {
e.printStackTrace();
} catch (SAXException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
}
}

运行结果:

SAX解析开始...
开始解析第1本书...
category:Java
title:Java多线程编程核心技术
author:高洪岩
year:2015
price:69.00
结束解析第1本书...
开始解析第2本书...
category:C++
title:Effective C++: 55 Specific Ways to Improve Your Programs and Designs
author:Scott Meyers
year:2006
price:58.00
结束解析第2本书...
开始解析第3本书...
category:Web
title:Learning XML
author:Erik T. Ray
year:2016
price:39.95
结束解析第3本书...
SAX解析结束...

4. 解析并储存于对象中

 
  
package com.qunar.bean;
 
/**
* book实体类
* @author sjf0115
*
*/
public class Book {
private String category;
private String title;
private String author;
private String year;
private String price;
private String lang;
public String getCategory() {
return category;
}
public void setCategory(String category) {
this.category = category;
}
public String getTitle() {
return title;
}
public void setTitle(String title) {
this.title = title;
}
public String getAuthor() {
return author;
}
public void setAuthor(String author) {
this.author = author;
}
public String getYear() {
return year;
}
public void setYear(String year) {
this.year = year;
}
public String getPrice() {
return price;
}
public void setPrice(String price) {
this.price = price;
}
public String getLang() {
return lang;
}
public void setLang(String lang) {
this.lang = lang;
}
@Override
public String toString() {
return "category:" + category + " lang:" + lang + " title:" + title + " author:" + author + " year:" + year + " price:" + price;
}
}

 
  
package com.qunar.handler;
 
import java.util.ArrayList;
import java.util.List;
 
import org.xml.sax.Attributes;
import org.xml.sax.SAXException;
import org.xml.sax.helpers.DefaultHandler;
 
import com.qunar.bean.Book;
 
 
public class SAXParserHandler extends DefaultHandler{
private Book book;
private int bookIndex = 0;
// 节点文本内容
private String text;
private List<Book> bookList = new ArrayList<Book>();
public List<Book> getBookList() {
return bookList;
}
// 用来标示解析开始
@Override
public void startDocument() throws SAXException {
System.out.println("SAX解析开始...");
}
// 用来标示解析结束
@Override
public void endDocument() throws SAXException {
System.out.println("SAX解析结束...");
}
// 用来遍历XML文件的开始标签
@Override
public void startElement(String uri, String localName, String qName, Attributes attributes) throws SAXException {
// 调用DefaultHandler类的startElement方法
super.startElement(uri, localName, qName, attributes);
// 开始解析book元素节点
if(qName.equals("book")){
// 创建一个book对象
book = new Book();
++ bookIndex;
System.out.println("开始解析第" + bookIndex + "本书...");
int size = attributes.getLength();
for(int i = 0;i < size;++i){
String attr = attributes.getQName(i);
// 属性category
if(attr.equals("category")){
book.setCategory(attributes.getValue(i));
}//if
}//for
}//if
// 用于遍历title节点中的属性
else if(qName.equals("title")){
int size = attributes.getLength();
for(int i = 0;i < size;++i){
String attr = attributes.getQName(i);
// 属性category
if(attr.equals("lang")){
book.setLang(attributes.getValue(i));
}//if
}//for
}//else
}
// 用来遍历XML文件的结束标签
@Override
public void endElement(String uri, String localName, String qName) throws SAXException {
super.endElement(uri, localName, qName);
// 判断一本书是否解析完
if(qName.equals("book")){
bookList.add(book);
book = null;
System.out.println("结束解析第" + bookIndex + "本书...");
}//if
else if(qName.equals("title")){
book.setTitle(text);
}//else
else if(qName.equals("author")){
book.setAuthor(text);
}//else
else if(qName.equals("year")){
book.setYear(text);
}//else
else if(qName.equals("price")){
book.setPrice(text);
}//else
}
// 文本值
@Override
public void characters(char[] ch, int start, int length) throws SAXException {
super.characters(ch, start, length);
text = new String(ch, start, length);
}
}

 
  
package com.qunar.xml;
 
import java.io.IOException;
import java.util.List;
 
import javax.xml.parsers.ParserConfigurationException;
import javax.xml.parsers.SAXParser;
import javax.xml.parsers.SAXParserFactory;
 
import org.xml.sax.SAXException;
 
import com.qunar.bean.Book;
import com.qunar.handler.SAXParserHandler;
 
 
/**
* SAX方式解析XML文档
* @author sjf0115
*
*/
public class SAXXMLCode {
 
public static void main(String[] args) {
String path = "D:\\bookstore.xml";
try {
// 通过SAXParserFactory的静态方法newInstance()方法获取SAXParserFactory实例对象factory
SAXParserFactory factory = SAXParserFactory.newInstance();
// 通过SAXParserFactory实例的newSAXParser()方法返回SAXParser实例parser
SAXParser saxParser = factory.newSAXParser();
// 定义SAXParserHandler对象
SAXParserHandler handler = new SAXParserHandler();
// 解析XML文档
saxParser.parse(path, handler);
// 得到遍历结果
List<Book> bookList = handler.getBookList();
System.out.println("遍历结果:");
for (Book book : bookList) {
System.out.println(book);
}//for
} catch (ParserConfigurationException e) {
e.printStackTrace();
} catch (SAXException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
}
}

运行结果:

SAX解析开始...
开始解析第1本书...
结束解析第1本书...
开始解析第2本书...
结束解析第2本书...
开始解析第3本书...
结束解析第3本书...
SAX解析结束...
遍历结果:
category:Java  lang:chi   title:Java多线程编程核心技术   author:高洪岩   year:2015   price:69.00
category:C++  lang:en   title:Effective C++: 55 Specific Ways to Improve Your Programs and Designs   author:Scott Meyers   year:2006   price:58.00
category:Web  lang:en   title:Learning XML   author:Erik T. Ray   year:2016   price:39.95



目录
相关文章
|
2月前
|
Java API Apache
Java编程如何读取Word文档里的Excel表格,并在保存文本内容时保留表格的样式?
【10月更文挑战第29天】Java编程如何读取Word文档里的Excel表格,并在保存文本内容时保留表格的样式?
187 5
|
3月前
|
开发框架 供应链 监控
并行开发模型详解:类型、步骤及其应用解析
在现代研发环境中,企业需要在有限时间内推出高质量的产品,以满足客户不断变化的需求。传统的线性开发模式往往拖慢进度,导致资源浪费和延迟交付。并行开发模型通过允许多个开发阶段同时进行,极大提高了产品开发的效率和响应能力。本文将深入解析并行开发模型,涵盖其类型、步骤及如何通过辅助工具优化团队协作和管理工作流。
114 3
|
4天前
|
人工智能 监控 数据可视化
提升开发效率:看板方法的全面解析
随着软件开发复杂度提升,并行开发模式下面临资源分配不均、信息传递延迟及缺乏全局视图等瓶颈问题。看板工具通过任务状态实时可视化、流量效率监控和任务依赖管理,帮助团队直观展示和解决这些瓶颈。未来,结合AI预测和自动化优化,看板工具将更高效地支持并行开发,成为驱动协作与创新的核心支柱。
|
18天前
|
JSON 供应链 搜索推荐
淘宝APP分类API接口:开发、运用与收益全解析
淘宝APP作为国内领先的购物平台,拥有丰富的商品资源和庞大的用户群体。分类API接口是实现商品分类管理、查询及个性化推荐的关键工具。通过开发和使用该接口,商家可以构建分类树、进行商品查询与搜索、提供个性化推荐,从而提高销售额、增加商品曝光、提升用户体验并降低运营成本。此外,它还能帮助拓展业务范围,满足用户的多样化需求,推动电商业务的发展和创新。
44 5
|
1月前
|
安全 前端开发 Android开发
探索移动应用与系统:从开发到操作系统的深度解析
在数字化时代的浪潮中,移动应用和操作系统成为了我们日常生活的重要组成部分。本文将深入探讨移动应用的开发流程、关键技术和最佳实践,同时分析移动操作系统的核心功能、架构和安全性。通过实际案例和代码示例,我们将揭示如何构建高效、安全且用户友好的移动应用,并理解不同操作系统之间的差异及其对应用开发的影响。无论你是开发者还是对移动技术感兴趣的读者,这篇文章都将为你提供宝贵的见解和知识。
|
2月前
|
存储 缓存 监控
后端开发中的缓存机制:深度解析与最佳实践####
本文深入探讨了后端开发中不可或缺的一环——缓存机制,旨在为读者提供一份详尽的指南,涵盖缓存的基本原理、常见类型(如内存缓存、磁盘缓存、分布式缓存等)、主流技术选型(Redis、Memcached、Ehcache等),以及在实际项目中如何根据业务需求设计并实施高效的缓存策略。不同于常规摘要的概述性质,本摘要直接点明文章将围绕“深度解析”与“最佳实践”两大核心展开,既适合初学者构建基础认知框架,也为有经验的开发者提供优化建议与实战技巧。 ####
|
1月前
|
Java 调度 Android开发
安卓与iOS开发中的线程管理差异解析
在移动应用开发的广阔天地中,安卓和iOS两大平台各自拥有独特的魅力。如同东西方文化的差异,它们在处理多线程任务时也展现出不同的哲学。本文将带你穿梭于这两个平台之间,比较它们在线程管理上的核心理念、实现方式及性能考量,助你成为跨平台的编程高手。
|
2月前
|
监控 前端开发 安全
如何开发一个网站:全面解析与实战指南
在数字化时代,网站是企业和个人展示形象、传播信息的关键平台。本文提供从规划、设计、开发、上线到后期维护的全方位网站开发指南,涵盖明确目标、分析用户、设定功能需求、设计风格、技术选型、测试部署及优化升级等内容,帮助你打造既美观又实用的网站。
129 4
|
2月前
|
前端开发 Android开发 UED
移动应用与系统:从开发到优化的全面解析####
本文深入探讨了移动应用开发的全过程,从最初的构思到最终的发布,并详细阐述了移动操作系统对应用性能和用户体验的影响。通过分析当前主流移动操作系统的特性及差异,本文旨在为开发者提供一套全面的开发与优化指南,确保应用在不同平台上均能实现最佳表现。 ####
44 0
|
2月前
|
监控 安全 Serverless
"揭秘D2终端大会热点技术:Serverless架构最佳实践全解析,让你的开发效率翻倍,迈向技术新高峰!"
【10月更文挑战第23天】D2终端大会汇聚了众多前沿技术,其中Serverless架构备受瞩目。它让开发者无需关注服务器管理,专注于业务逻辑,提高开发效率。本文介绍了选择合适平台、设计合理函数架构、优化性能及安全监控的最佳实践,助力开发者充分挖掘Serverless潜力,推动技术发展。
103 1

推荐镜像

更多