Dom4j  是一个易用的、开源的库,用于 XML ,XPath  和 XSLT 。它应用于 Java  平台,采用了 Java  集合框架并完全支持 DOM ,SAX 和 和 JAXP 。我们可以很方便的使用DOM4J来对一个XML进行CRUD操作,当然也可以面向对象编程,将一个XML里面的数据来转换成相关对象使用。

  • 常用有4个接口,4个工具类:

Node  Node:为所有的 dom4j 中 XML 节点定义了多态行为

Element  Element:定义 XML 元素

Attribute  Attribute:定义了 XML 的属性

Text  Text:定义 XML 文本节点。

SAXReader:sax解析工具类

DOMReader:dom解析工具类

XMLWriter:写入XML文档工具类

OutputFormat:输入格式化工具类。

  • 值得注意的2点是:

1,格式化输出和指定编码。默认的输出方式为紧凑方式,默认编码为 UTF-8,但对于我们的应用而言,一般都要用到中文,并且希望显示时按自动缩进的方式的显示,这就需用到OutputFormat 类。

2,Dom4j 编码问题。j ava中由W ri t er类继承下来的子类没有提供编码格式处理,所以dom 4j 也就无法对输出的文件进行正确的格式处理。这时候所保存的文件会以系统的默认编码对文件进行保存,在中文版的wi ndow下j ava的默认的编码为G BK,也就是所虽然我们标识了要将xm l 保存为ut f -8格式但实际上文件是以G BK格式来保存的,所以这也就是为什么能够我们使用G BK、G B2312编码来生成xm
l 文件能正确的被解析,而以U TF-8格式生成的文件不能被xm l 解析器所解析的原因。所以在创建XMLWriter这个对象时,最好使用FileOutputStream,不要用FileWriter。



以下代码使用了DOM4J对XML实现了CRUD操作:

import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.FileWriter;
import java.io.IOException;
import java.io.UnsupportedEncodingException;
import java.net.URL;
import java.util.List; import org.dom4j.Attribute;
import org.dom4j.Comment;
import org.dom4j.Document;
import org.dom4j.DocumentException;
import org.dom4j.DocumentHelper;
import org.dom4j.Element;
import org.dom4j.Node;
import org.dom4j.Text;
import org.dom4j.io.OutputFormat;
import org.dom4j.io.SAXReader;
import org.dom4j.io.XMLWriter; /**
*
* @version 1L
* @author LinkinPark
* @since 2014-12-29
* @motto 梦似烟花心似水,同学少年不言情
* @desc ^其实DOM4J还是比较简单的,API也不是很多,着重记住Document,Element,Attribute就好了。
* Node是上面所有的接口的公共一个多态接口
*/
public class Dom4jHandler
{
public Document parse(URL url) throws DocumentException
{
SAXReader reader = new SAXReader();
Document document = reader.read(url);
return document;
} //解析XML
public void read(String fileName) throws DocumentException
{
SAXReader reader = new SAXReader();
//定义XML文档
Document document = reader.read(this.getClass().getResourceAsStream("/" + fileName));
//定义XML元素
Element rootElement = document.getRootElement();
System.out.println("根节点是:" + rootElement.getName());
List<Element> childElements = rootElement.elements();
for (Element element : childElements)
{
System.out.println(element.getName());
//下面这个nodeCount里面包含了所有的节点,那个换行也包含上了呢
System.out.println(element.nodeCount());
for (int i = 0; i < element.nodeCount(); i++)
{
Node node = element.node(i);
System.out.println("节点的名字是" + node.getName() + ";节点的值是" + node.getText() + ";节点的类型是" + node.getNodeTypeName());
if (node instanceof Element)
{
System.out.println("这里是元素!");
}
else if (node instanceof Comment)
{
System.out.println("这里是注释!");
}
else
{
System.out.println("这里鸡毛都没有!");
}
}
//现在开始解决上面那个空白的换行问题 如何去掉呢?
System.out.println("========开始去掉那个空白============");
for (int i = 0; i < element.nodeCount(); i++)
{
Node node = element.node(i);
if (node instanceof Text)
{
System.out.println("这里是文本!");
if ("".equals(node.getText().trim()))
{
System.out.println("这里就是要去掉的空白吆。。。");
}
else
{
System.out.println("这里是内容不是空白,所以不要去掉。。。");
System.out.println(node.getText().trim());
}
}
}
System.out.println("=========去掉空白结束=============");
List<Attribute> attributes = element.attributes();
//遍历这个元素里面的所有的属性
for (Attribute attribute : attributes)
{
System.out.println(attribute.getName() + ":" + attribute.getValue());
}
//现在我只想得到version这个属性
Attribute attribute = element.attribute("version");
System.out.println(attribute.getName() + ":" + attribute.getValue());
//现在我只想直接拿出version这个属性的值
String attributeValue = element.attributeValue("version");
System.out.println(attributeValue);
//获得element这个元素里面的所有的子节点
List<Element> childs = element.elements();
for (Element element2 : childs)
{
//Element元素里面没有getValue方法,Attribute里面有,其实和getText是一样的
System.out.println(element2.getName() + ":" + element2.getText());
}
//同样的Element也可以直接获得值,而且还可以去除空格
System.out.println(element.elementText("driver"));
System.out.println(element.elementTextTrim("driver"));
}
} //写出一个XML
public void write(String fileName) throws Exception
{
//生成一个XML文档
Document document = DocumentHelper.createDocument();
//给XML文档也就是Document添加数据
Element root = document.addElement("DataSource");//添加根节点
root.addComment("这里是LinkinPark自己生成的XML。。。");//添加注释
Element dateBase = root.addElement("database");//在root根节点下添加一个子节点
dateBase.addAttribute("name", "mysql");//给dateBase添加属性
dateBase.addAttribute("version", "5.0");//给dateBase添加属性
dateBase.addElement("driver").setText("com.mysql.jdbc.Driver");//给dateBase添加子节点
dateBase.addElement("url").setText("jdbc:mysql://localhost:3306/linkinjdbc");//给dateBase添加子节点
dateBase.addElement("user").setText("root");//给dateBase添加子节点
dateBase.addElement("password").setText("root");//给dateBase添加子节点
//将这个XML文档写出到文件去
writerTo(document, fileName);
} //专门定义一个写出XML到指定文件的方法
public void writerTo(Document document, String fileName) throws Exception
{
//OutputFormat of = OutputFormat.createCompactFormat();//这个format是不换行的,没有格式化过的
OutputFormat of = OutputFormat.createPrettyPrint();//这里是格式化过的
of.setEncoding("UTF-8");
//XMLWriter writer = new XMLWriter(new FileWriter(fileName), of);
//最好使用下面这种情况,这样子可以有效的解决XML编码是UTF-8的编码错误问题
XMLWriter writer = new XMLWriter(new FileOutputStream(fileName), of);
writer.write(document);
writer.close();
} //修改一个XML 2个参数:一个是读取文件自己的位置,一个是要保存到的文件的位置。
//这里也很好的说明了加载文件和流的区别:使用类加载器是直接从内存中要的,所以相对路径没有src;但是要是使用一个流来指定输入位置,那么就是文件系统,src要有的。
public void update(String fileName1, String fileName2) throws Exception
{
//现在我要修改LinkinPark下面的数据库的名称和数据库连接的名称,一个是属性,一个是节点
SAXReader reader = new SAXReader();
Document document = reader.read(this.getClass().getResourceAsStream("/" + fileName1));
Element rootElement = document.getRootElement();
List<Element> childElements = rootElement.elements("database");
for (Element element : childElements)
{
if ("mysql".equals(element.attributeValue("name")))
{
//换了这个属性的名字
element.attribute("name").setText("Oracle1");
}
List<Element> childs = element.elements("driver");
for (Element element2 : childs)
{
System.out.println(element2.getText().trim());
if ("com.mysql.jdbc.Driver".equals(element2.getText().trim()))
{
element2.setText("oracle.jdbc.driver.OracleDriver");
}
}
}
writerTo(document, fileName2);
} //删除一个XML里面的一部分内容
public void delete(String fileName1, String fileName2) throws Exception
{
//现在我要删除LinkinPark下面的数据库的名称和数据库连接的名称,一个是属性,一个是节点
SAXReader reader = new SAXReader();
Document document = reader.read(this.getClass().getResourceAsStream("/" + fileName1));
Element rootElement = document.getRootElement();
List<Element> childElements = rootElement.elements("database");
for (Element element : childElements)
{
//删除掉这个属性
element.remove(element.attribute("name"));
//删除掉这个子节点
element.remove(element.element("driver"));
}
writerTo(document, fileName2);
} public static void main(String[] args) throws Exception
{
//new Dom4jHandler().read("data-sources.xml");
new Dom4jHandler().write("src/LinkinPark...");
//new Dom4jHandler().update("LinkinPark...", "src/LinkinPark...");
//new Dom4jHandler().delete("LinkinPark...", "src/LinkinPark...");
} }
<?xml version="1.0" encoding="UTF-8"?>

<DataSource>
<!--这里是LinkinPark自己生成的XML。。。-->
<database name="mysql" version="5.0">
<driver>com.mysql.jdbc.Driver</driver>
<url>jdbc:mysql://localhost:3306/linkinjdbc</url>
<user>root</user>
<password>root</password>
</database>
</DataSource>

无废话XML--DOM4J的更多相关文章

  1. 无废话XML--XML细节

    今天开始研究xml,其实在实际的开发中,我们参与到真正的XML开发并不是很多,最多写一个配置,但是我还是觉得很有必要把XML的知识整理一遍.作为基本的2种的数据交互载体(还有一个是json),基本的X ...

  2. WCF入门教程(四)通过Host代码方式来承载服务 一个WCF使用TCP协议进行通协的例子 jquery ajax调用WCF,采用System.ServiceModel.WebHttpBinding System.ServiceModel.WSHttpBinding协议 学习WCF笔记之二 无废话WCF入门教程一[什么是WCF]

    WCF入门教程(四)通过Host代码方式来承载服务 Posted on 2014-05-15 13:03 停留的风 阅读(7681) 评论(0) 编辑 收藏 WCF入门教程(四)通过Host代码方式来 ...

  3. 无废话ExtJs 入门教程二十一[继承:Extend]

    无废话ExtJs 入门教程二十一[继承:Extend] extjs技术交流,欢迎加群(201926085) 在开发中,我们在使用视图组件时,经常要设置宽度,高度,标题等属性.而这些属性可以通过“继承” ...

  4. 无废话ExtJs 入门教程二十[数据交互:AJAX]

    无废话ExtJs 入门教程二十[数据交互:AJAX] extjs技术交流,欢迎加群(521711109) 1.代码如下: 1 <!DOCTYPE html PUBLIC "-//W3C ...

  5. 无废话ExtJs 入门教程十九[API的使用]

    无废话ExtJs 入门教程十九[API的使用] extjs技术交流,欢迎加群(201926085) 首先解释什么是 API 来自百度百科的官方解释:API(Application Programmin ...

  6. 无废话ExtJs 入门教程十七[列表:GridPanel]

    无废话ExtJs 入门教程十七[列表:GridPanel] extjs技术交流,欢迎加群(201926085) 在Extjs中,GridPanel用于数据显示,即我们平时说的列表页.在本节中,我们先对 ...

  7. 无废话WCF入门教程六[一个简单的Demo]

    一.前言 前面的几个章节介绍了很多理论基础,如:什么是WCF.WCF中的A.B.C.WCF的传输模式.本文从零开始和大家一起写一个小的WCF应用程序Demo. 大多框架的学习都是从增.删.改.查开始来 ...

  8. 无废话ExtJs 入门教程十六[页面布局:Layout]

    无废话ExtJs 入门教程十六[页面布局:Layout] extjs技术交流,欢迎加群(201926085) 首先解释什么是布局: 来自百度词典的官方解释:◎ 布局 bùjú: [distributi ...

  9. 无废话ExtJs 入门教程十五[员工信息表Demo:AddUser]

    无废话ExtJs 入门教程十五[员工信息表Demo:AddUser] extjs技术交流,欢迎加群(201926085) 前面我们共介绍过10种表单组件,这些组件是我们在开发过程中最经常用到的,所以一 ...

  10. 无废话ExtJs 入门教程十四[文本编辑器:Editor]

    无废话ExtJs 入门教程十四[文本编辑器:Editor] extjs技术交流,欢迎加群(201926085) ExtJs自带的编辑器没有图片上传的功能,大部分时候能够满足我们的需要. 但有时候这个功 ...

随机推荐

  1. python3之面向对象

    1.面向对象术语 类(Class): 用来描述具有相同的属性和方法的对象的集合.它定义了该集合中每个对象所共有的属性和方法.对象是类的实例. 类属性(类变量):类属性在整个实例化的对象中是公用的.类属 ...

  2. [已解决]IndentationError: unindent does not match any outer indentation level

    最近跟同事合作代码没有用git进行协同,很多代码拷贝后进行粘贴,常常报以上错误. 经过查询发现是空格跟tab混合使用了,重新将代码的缩进手动调整下就好了.

  3. React Native随笔——组件TextInput

    一.实例 先看一下我要做的搜索框的样子 需要一个Image,和一个TextInput 去掉默认下划线 underlineColorAndroid='transparent' 设置光标颜色 select ...

  4. 关于soapui简介与入门

    SoapUI简介 SoapUI是一个开源测试工具,通过soap/http来检查.调用.实现Web Service的功能/负载/符合性测试.该工具既可作为一个单独的测试软件使用,也可利用插件集成到Ecl ...

  5. 2017"百度之星"程序设计大赛 - 资格赛【1001 Floyd求最小环 1002 歪解(并查集),1003 完全背包 1004 01背包 1005 打表找规律+卡特兰数】

    度度熊保护村庄 Accepts: 13 Submissions: 488 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/3276 ...

  6. ElasticSearch + xpack 使用.md

    ElasticSearch 是一个高可用开源全文检索和分析组件.提供存储服务,搜索服务,大数据准实时分析等.一般用于提供一些提供复杂搜索的应.我们为什么要选择 ElasticSearch ?因为它是一 ...

  7. c语言基础学习02_windows系统下的cmd命令

    =============================================================================注意:cmd的命令很多,需要用的时候可以查询即 ...

  8. [国嵌攻略][151][nandflash驱动程序设计]

    初始化 打开/drivers/mtd/nand/s3c2410.c找到nand flash驱动程序代码,找到模块初始化函数s3c_nand_init,找到platform_driver中的probe函 ...

  9. [国嵌攻略][097][U-Boot新手入门]

    嵌入式软件层次 1.Bootloader 2.Linux内核 3.文件系统 编译U-Boot 1.解压uboot tar zxvf uboot.tar.gz 2.清除uboot make distcl ...

  10. Spider_Man_5.2 の Mongodb_使用

    一:简介 MongoDB是一款强大.灵活.且易于扩展的通用型数据库1.易用性 MongoDB是一个面向文档(document-oriented)的数据库,而不是关系型数据库.不采用关系型主要是为了获得 ...