Java中如何引用文档对象模型_在JAVA中使用文档对象模型DOM经验小结

文档对象模型 (DOM) 是一个文档标准,对于完备的文档和复杂的应用程序,DOM 提供了大量灵活性。DOM标准是标准的。它很强壮且完整,并且有许多实现。这是许多大型安装的决定因素--特别是对产品应用程序,以避免在API发生改变时进行大量的改写。

以上是我在选择处理XML数据时之所以没有选择JDOM或者dom4j等其它面向对象的标准的原因,不过也由于DOM从一开始就是一种与语言无关的模型,而且它更趋向用于像C或Perl这类语言,没有利用Java的面向对象的性能,所以在使用的过程中也遇到了不少的麻烦,今天这里做一个小结。另外,我目前使用XML主要是作为数据传输的统一格式,并统一用户界面展示的接口,应用的面并不是很广,所以使用到的DOM的内容其实不多。

在准备使用它的时候,是做了充足的准备的,也有遇到困难的准备,所以一开始就有了一个简单的工具类来封装DOM对象使用时必要的公共方法,实际证明这样做是很明智的,一个简单的创建Document对象的操作,要是每次都需要写上5行以上代码,并且还要处理那些烦人的Exception,实在是会打击大家的积极性,所以在最初,做了一个XMLTool类,专门封装了如下的公共方法:

1、 Document对象创建(包括空的Document对象创建,以一个给定Node节点作为根节点创建。

2、 将一个规范的XML字符串转换成一个Document对象。

3、 从物理硬盘读取一个XML文件并返回一个Document对象。

4、 将一个Node对象转换成字符串。

其中每个方法都截获相关的DOM操作所抛出的异常,转换成一个RuntimeException抛出,这些异常在实际使用过程中,一般状况下其实都不会抛出,特别是象生成一个Document对象时的ParserConfigurationException、转换Node节点成字符串时要生成一个Transformer对象时的TransformerConfigurationException等等,没有必要在它们身上花时间精力。而且真就出了相关的异常的话,其实根本没有办法处理,这样的状况通常是系统环境配置有问题(比如必要的DOM实现解析器等包没有加入环境),所以包装该异常时只是很简要的获取其Message抛出。

代码如下:

/**

* 初始化一个空Document对象返回。

* @return a Document

*/

public static Document newXMLDocument() {

try {

return newDocumentBuilder().newDocument();

} catch (ParserConfigurationException e) {

throw new RuntimeException(e.getMessage());

}

}

/**

* 初始化一个DocumentBuilder

* @return a DocumentBuilder

* @throws ParserConfigurationException

*/

public static DocumentBuilder newDocumentBuilder()

throws ParserConfigurationException {

return newDocumentBuilderFactory().newDocumentBuilder();

}

/**

* 初始化一个DocumentBuilderFactory

* @return a DocumentBuilderFactory

*/

public static DocumentBuilderFactory newDocumentBuilderFactory() {

DocumentBuilderFactory dbf = DocumentBuilderFactory.newInstance();

dbf.setNamespaceAware(true);

return dbf;

}

/**

* 将传入的一个XML String转换成一个org.w3c.dom.Document对象返回。

* @param xmlString 一个符合XML规范的字符串表达。

* @return a Document

*/

public static Document parseXMLDocument(String xmlString) {

if (xmlString == null) {

throw new IllegalArgumentException();

}

try {

return newDocumentBuilder().parse(

new InputSource(new StringReader(xmlString)));

} catch (Exception e) {

throw new RuntimeException(e.getMessage());

}

}

/**

* 给定一个输入流,解析为一个org.w3c.dom.Document对象返回。

* @param input

* @return a org.w3c.dom.Document

*/

public static Document parseXMLDocument(InputStream input) {

if (input == null) {

throw new IllegalArgumentException("参数为null!");

}

try {

return newDocumentBuilder().parse(input);

} catch (Exception e) {

throw new RuntimeException(e.getMessage());

}

}

/**

* 给定一个文件名,获取该文件并解析为一个org.w3c.dom.Document对象返回。

* @param fileName 待解析文件的文件名

* @return a org.w3c.dom.Document

*/

public static Document loadXMLDocumentFromFile(String fileName) {

if (fileName == null) {

throw new IllegalArgumentException("未指定文件名及其物理路径!");

}

try {

return newDocumentBuilder().parse(new File(fileName));

} catch (SAXException e) {

throw new IllegalArgumentException(

"目标文件(" + fileName + ")不能被正确解析为XML!\n" + e.getMessage());

} catch (IOException e) {

throw new IllegalArgumentException(

"不能获取目标文件(" + fileName + ")!\n" + e.getMessage());

} catch (ParserConfigurationException e) {

throw new RuntimeException(e.getMessage());

}

}

/**

* 给定一个节点,将该节点加入新构造的Document中。

* @param node a Document node

* @return a new Document

*/

public static Document newXMLDocument(Node node) {

Document doc = newXMLDocument();

doc.appendChild(doc.importNode(node, true));

return doc;

}

/**

* 将传入的一个DOM Node对象输出成字符串。如果失败则返回一个空字符串""。

* @param node DOM Node 对象。

* @return a XML String from node

*/

public static String toString(Node node) {

if (node == null) {

throw new IllegalArgumentException();

}

Transformer transformer = newTransformer();

if (transformer != null) {

try {

StringWriter sw = new StringWriter();

transformer.transform(

new DOMSource(node),

new StreamResult(sw));

return sw.toString();

} catch (TransformerException te) {

throw new RuntimeException(te.getMessage());

}

}

return errXMLString("不能生成XML信息!");

}

/**

* 将传入的一个DOM Node对象输出成字符串。如果失败则返回一个空字符串""。

* @param node DOM Node 对象。

* @return a XML String from node

*/

public static String toString(Node node) {

if (node == null) {

throw new IllegalArgumentException();

}

Transformer transformer = newTransformer();

if (transformer != null) {

try {

StringWriter sw = new StringWriter();

transformer.transform(

new DOMSource(node),

new StreamResult(sw));

return sw.toString();

} catch (TransformerException te) {

throw new RuntimeException(te.getMessage());

}

}

return errXMLString("不能生成XML信息!");

}

/**

* 获取一个Transformer对象,由于使用时都做相同的初始化,所以提取出来作为公共方法。

* @return a Transformer encoding gb2312

*/

public static Transformer newTransformer() {

try {

Transformer transformer =

TransformerFactory.newInstance().newTransformer();

Properties properties = transformer.getOutputProperties();

properties.setProperty(OutputKeys.ENCODING, "gb2312");

properties.setProperty(OutputKeys.METHOD, "xml");

properties.setProperty(OutputKeys.VERSION, "1.0");

properties.setProperty(OutputKeys.INDENT, "no");

transformer.setOutputProperties(properties);

return transformer;

} catch (TransformerConfigurationException tce) {

throw new RuntimeException(tce.getMessage());

}

}

/**

* 返回一段XML表述的错误信息。提示信息的TITLE为:系统错误。之所以使用字符串拼装,主要是这样做一般

* 不会有异常出现。

* @param errMsg 提示错误信息

* @return a XML String show err msg

*/

public static String errXMLString(String errMsg) {

StringBuffer msg = new StringBuffer(100);

msg.append("<?xml version=\"1.0\" encoding=\"gb2312\" ?>");

msg.append("");

return msg.toString();

}

/**

* 返回一段XML表述的错误信息。提示信息的TITLE为:系统错误

* @param errMsg 提示错误信息

* @param errClass 抛出该错误的类,用于提取错误来源信息。

* @return a XML String show err msg

*/

public static String errXMLString(String errMsg, Class errClass) {

StringBuffer msg = new StringBuffer(100);

msg.append("<?xml version=\"1.0\" encoding=\"gb2312\" ?>");

msg.append(

"

+ errMsg

+ "\" errSource=\""

+ errClass.getName()

+ "\"/>");

return msg.toString();

}

/**

* 返回一段XML表述的错误信息。

* @param title 提示的title

* @param errMsg 提示错误信息

* @param errClass 抛出该错误的类,用于提取错误来源信息。

* @return a XML String show err msg

*/

public static String errXMLString(

String title,

String errMsg,

Class errClass) {

StringBuffer msg = new StringBuffer(100);

msg.append("<?xml version=\"1.0\" encoding=\"gb2312\" ?>");

msg.append(

"

+ title

+ "\" errMsg=\""

+ errMsg

+ "\" errSource=\""

+ errClass.getName()

+ "\"/>");

return msg.toString();

}

以上都是DOM的基本应用,所以就不一一详细说明了。

在实际使用过程中,有几种状况使用很频繁,但是DOM的接口的设计却使该操作很麻烦,所以分别添加了相应的处理方法。

其中最麻烦的要数获取一个节点的Text子节点文本信息了,如下的XML节点:

text

在拥有element节点对象时,要获取其中的文本信息"text",首先要获取element节点的子节点列表,要判断其是否存在子节点,如果存在,那么遍历其子节点找到一个TextNode节点,通过getNodeValue()方法来获取该文本信息,由于这里element节点没有信息时没有子节点,所以必须判断element节点是否存在子节点才能去访问真正包含了文本信息的TextNode节点,那么如果要处理的数据都是以这种形式给出的,就会增加大量的开发代码同时让开发工作枯燥无味,因此这里使用了一个默认的约定实现,就是,给出了一个公共方法,该方法取给定Node下的直接子节点的Text节点文本信息,如果不存在Text节点则返回null,这个约定虽然使该方法的使用有所限制,也可能导致错误使用该方法,但是,按实际使用的状况来看,这样的约定和使用方式是没有问题的,因为实际用到的都是上面举的例子的状况,代码:

/**

* 这个方法获取给定Node下的Text节点文本信息,如果不存在Text节点则返回null。

* 注意:是直接子节点,相差2层或2层以上不会被考虑。

* @param node a Node 一个Node。

* @return a String 如果给定节点存在Text子节点,则返回第一个访问到的Text子节点文本信息,如果不存在则返回null。

*/

public static String getNodeValue(Node node) {

if (node == null) {

return null;

}

Text text = getTextNode(node);

if (text != null) {

return text.getNodeValue();

}

return null;

}

/**

* 这个方法获取给定Node下的Text节点,如果不存在Text节点则返回null。

* 注意:是直接子节点,相差2层或2层以上不会被考虑。

* @param node a Node 一个Node。

* @return a Text 如果给定节点存在Text子节点,则返回第一个访问到的Text子节点,如果不存在则返回null。

*/

public static Text getTextNode(Node node) {

if (node == null) {

return null;

}

if (node.hasChildNodes()) {

NodeList list = node.getChildNodes();

for (int i = 0; i < list.getLength(); i++) {

if (list.item(i).getNodeType() == Node.TEXT_NODE) {

return (Text) list.item(i);

}

}

}

return null;

}

上面代码将获取给定Node节点的直接Text子节点分开包装。

另一个很经常碰到的状况是,我希望直接定位到目标节点,获取该节点对象,而不需要通过一层一层的节点遍历来找到目标节点,DOM2接口中至少提供了如下的方式来定位节点:

1、 对于Document对象:

1) getDocumentElement()

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/245055.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

禅道项目管理_禅道 11.6.1 版本发布,完善细节,修复 Bug

禅道项目管理软件集产品管理、项目管理、质量管理、文档管理、组织管理和事务管理于一体&#xff0c;是一款功能完备的项目管理软件&#xff0c;完美地覆盖了项目管理的核心流程。禅道官网&#xff1a;www.zentao.net。大家好&#xff0c;禅道项目管理软件11.6.1发布&#xff0…

mendeley引用参考文献不显示_免费文献管理器Mendeley

June 2020有机合成化学文献检索今天小编给大家分享一款免费又好用的文献管理器——Mendeley&#xff0c;另外晶体cif文件下载—Materialsproject和COD数据库可在菜单栏的文献检索[文献管理/资源]中查看Mendeley是什么Mendeley是一款免费的跨平台文献管理软件&#xff0c;同时也…

pgsql vs mysql查询_对比平台--SQL Server Vs PostgreSQL

Microsoft SQL Server是一个数据库管理和分析系统&#xff0c;主要用于电子商务&#xff0c;业务范围和不同的数据仓库解决方案。另一方面&#xff0c;PostgreSQL是高级的对象关系数据库管理系统&#xff0c;它为SQL标准的扩展子集提供支持&#xff0c;包括不同的事务&#xff…

停车场管理系统代码_jsp19109商场商铺停车场服务系统-SSM-Mysql

jsp19109商场商铺停车场服务系统-SSM-Mysql该设计有演示视频    100%能运行买重包换  保密发送  一校一份编号&#xff1a;jsp19109语言数据库&#xff1a;jspMysql论文字数&#xff1a;12032字摘 要随着社会的发展&#xff0c;社会的方方面面都在利用信息化时代的优势。计…

java中compare语句的用法_Java RuleBasedCollator compare()用法及代码示例

java.text.RuleBasedCollat​​or类的compare()方法用于比较两个对象的强度&#xff0c;并且根据结果将返回0&#xff0c;正值和负值作为输出。用法:public int compare(Object o1, Object o2)参数&#xff1a;此方法需要两个对象之间进行比较。返回值&#xff1a;如果第一个对…

qregexp限制数字范围_数字系统实现电压电流控制的必经之路数模转换器

《芯势力》系列接上一篇文章&#xff0c;我们了解到了模数转换器&#xff0c;本文将带你了解数模转换器。看名字就能知道&#xff0c;如果模数转换器实现了模拟信号到数字信号的转换&#xff0c;那么&#xff0c;数模转换器就是模数转换器的逆过程&#xff0c;即把数字信号转换…

java 什么时候依赖注入_玩框架java依赖注入 – 何时使用单例

So I am wondering, should I be using singleton objects as the examples seem to imply? If this is the case, what is the advantage compared to the old static methods approach?依赖注入是一种将应用程序连接在一起的技术.您编写的组件并不直接相互依赖.而是将组件注…

四天人工智能 python入门体验课_百度深度学习7天打卡营,用Python+AI识别“青你2”小姐姐的高颜值...

原标题&#xff1a;百度深度学习7天打卡营&#xff0c;用PythonAI识别“青你2”小姐姐的高颜值“淡黄的长裙&#xff0c;蓬松的头发”&#xff0c;一夜之间洗脑全网&#xff0c;小姐姐们实在太让人上头了&#xff01;导师“小甜豆”Lisa 满屏的大长腿、蚂蚁腰&#xff0c;又飒又…

js方式调用php_js如何调用php函数

js调用php函数的方法&#xff1a;jQuery.ajax({type: "POST",url: your_functions_address.php,dataType: json,data: {functionname: add, arguments: [1, 2]},success: function (obj, textstatus) {if( !(error in obj) ) {yourVariable obj.result;}else {conso…

最大子序列求和_算法——求最大子段和

一、问题描述给定由n个整数组成的序列(a_1,a_2,…,a_n)&#xff0c;最大子段和问题要求该序列形如 的最大值(1≤i≤j≤n)&#xff0c;当序列中所有整数均为负整数时&#xff0c;其最大子段和为0。例如&#xff0c;序列(-20, 11, -4, 13, -5, -2)的最大子段和为&#xff1a; 注意…

seo黑帽劫持用的php,黑帽seo 论坛:黑帽seo防止网站被k的js劫持跳转代码

由于目前百度搜索百度搜索引擎对于js代码还没有办法完全辨别&#xff0c;因此也就出现了运用js代码跳转的黑帽优化提升手法。现如今在网络上有关js跳转代码不计其数&#xff0c;但是作为黑帽优化提升的seo手法之一&#xff0c;如何确保有效降低跳转的网址被k危害性&#xff0c;…

python vtk mousemove_VTK的视点研究之三维空间漫游(转载)

VTK的视点研究之三维空间漫游(转载)分类&#xff1a;计算机2009-08-17 16:19阅读(?)评论(0)#include #include "vtkConeSource.h"#include "vtkPolyDataMapper.h"#include "vtkRenderWindow.h"#include "vtkCamera.h"#include "…

php实现金币提现,PHP实现微信提现功能

本文实例为大家分享了PHP实现微信提现功能的具体代码&#xff0c;供大家参考&#xff0c;具体内容如下一、实现功能这几天在小程序里要实现用户从系统中提现到零钱的功能&#xff0c;查了一下文档可以使用 企业付款到用户零钱 来实现&#xff1b;官方文档注意事项&#xff1a;商…

oracle 同义词_【干货7】Oracle知识关键代码摘要

&#xff08;如果我分享的干货内容对你有帮助&#xff0c;可以通过赞或者评论的方式告诉我&#xff0c;我会持续分享&#xff1b;或者留言你想要的IT方面的支持&#xff0c;我将分享大家感兴趣的IT类技术干货&#xff1b;如果没有收到大家的反馈&#xff0c;10天后我将停止技术…

php new static,PHP面向对象中new self( )和 new static( ) 的区别

首先阐明结论&#xff0c;在PHP中 self指向定义了当前被调用方法的类&#xff0c; static指向调用当前静态方法的类。接下来通过一个例子来证明上面的结果class A{public static $_a Class A;public static function echoProperty(){echo self::$_a . PHP_EOL;}}class B exten…

python 动态规划_DP动态规划(Python实现)

前言_我们遇到的问题中&#xff0c;有很大一部分可以用动态规划(简称DP)来解。 解决这类问题可以很大地提升你的能力与技巧&#xff0c;我会试着帮助你理解如何使用DP来解题。 这篇文章是基于实例展开来讲的&#xff0c;因为干巴巴的理论实在不好理解。注意&#xff1a;如果你对…

php gif 透明,解决PHP剪切缩略图生成png,gif透明图时,黑色背景问题

背景图填充白色背景$white imagecolorallocate($dstim,255,255,255);imagefilledrectangle($dstim,0,0,$width,$height,$white);imagecolortransparent($dstim,$white);设置图片走透明通道$img imagecreatefrompng($src);imagesavealpha($img,true);//这里很重要;$thumb ima…

qt做的接收串口数据并显示曲线_QT无人机地面站设计与制作

近年来&#xff0c;无人机可谓是大火。无论是军事&#xff0c;还是民用&#xff0c;它的地位更是不用说。但&#xff0c;如何利用利用现有技术对无人机的信息进行操作&#xff0c;实现人、机合一呢&#xff1f;“无人机地面站”应运而生&#xff0c;结合仿真系统为地面工作人员…

php直接读取csv文件,php实现的读取CSV文件函数示例

本文实例讲述了php实现的读取CSV文件函数。分享给大家供大家参考&#xff0c;具体如下&#xff1a;function read_csv($cvs) {$shuang false;$str file_get_contents($cvs);for ($i0;$iif($str{$i}") {if($shuang) {if($str{$i1}") {$str{$i} *;$str{$i1} *;} el…

系统背景描述_【计算机论文】管件加工管理系统和数据库的结构探析

摘 要:结合"中国制造2025"及德国"工业4.0"的发展趋势,概述目前国内管件生产加工流程的现状和不足,基于对管件加工过程中管件之间的差别、管件加工批次的混合等特点导致的管理难点分析,介绍管件生产加工管理系统的设计思路和工作流程,并对该系统未来可进一步…