java Dom4j处理XML
一、前言
dom4j是一套非常优秀的java开源api,主要用于读写xml文档,具有性能优异、功能强大、和非常方便使用的特点。 另外xml经常用于数据交换的载体,像调用webservice传递的参数,以及数据做同步操作等等, 所以使用dom4j解析xml是非常有必要的。
二、准备条件
dom4j.jar
下载地址:http://sourceforge.net/projects/dom4j/
三、使用Dom4j实战
1、解析xml文档
实现思路:
<1>根据读取的xml路径,传递给SAXReader之后 返回一个Document文档对象;
<2>然后操作这个Document对象,获取下面的节点以及子节点的信息;
具体代码如下:
- import java.io.File;
- import java.io.FileInputStream;
- import java.io.InputStream;
- import java.util.Iterator;
- import java.util.List;
- import org.dom4j.Document;
- import org.dom4j.DocumentHelper;
- import org.dom4j.Element;
- import org.dom4j.io.SAXReader;
- /**
- * 使用dom4j解析xml文档
- * @author Administrator
- *
- */
- public class Dom4jParseXmlDemo {
- public void parseXml01(){
- try{
- //将src下面的xml转换为输入流
- InputStream inputStream = new FileInputStream(new File("D:/project/dynamicWeb/src/resource/module01.xml"));
- //InputStream inputStream = this.getClass().getResourceAsStream("/module01.xml");//也可以根据类的编译文件相对路径去找xml
- //创建SAXReader读取器,专门用于读取xml
- SAXReader saxReader = new SAXReader();
- //根据saxReader的read重写方法可知,既可以通过inputStream输入流来读取,也可以通过file对象来读取
- //Document document = saxReader.read(inputStream);
- Document document = saxReader.read(new File("D:/project/dynamicWeb/src/resource/module01.xml"));//必须指定文件的绝对路径
- //另外还可以使用DocumentHelper提供的xml转换器也是可以的。
- //Document document = DocumentHelper.parseText("<?xml version=\"1.0\" encoding=\"UTF-8\"?><modules id=\"123\"><module> 这个是module标签的文本信息</module></modules>");
- //获取根节点对象
- Element rootElement = document.getRootElement();
- System.out.println("根节点名称:" + rootElement.getName());//获取节点的名称
- System.out.println("根节点有多少属性:" + rootElement.attributeCount());//获取节点属性数目
- System.out.println("根节点id属性的值:" + rootElement.attributeValue("id"));//获取节点的属性id的值
- System.out.println("根节点内文本:" + rootElement.getText());//如果元素有子节点则返回空字符串,否则返回节点内的文本
- //rootElement.getText() 之所以会换行是因为 标签与标签之间使用了tab键和换行符布局,这个也算是文本所以显示出来换行的效果。
- System.out.println("根节点内文本(1):" + rootElement.getTextTrim());//去掉的是标签与标签之间的tab键和换行符等等,不是内容前后的空格
- System.out.println("根节点子节点文本内容:" + rootElement.getStringValue()); //返回当前节点递归所有子节点的文本信息。
- //获取子节点
- Element element = rootElement.element("module");
- if(element != null){
- System.out.println("子节点的文本:" + element.getText());//因为子节点和根节点都是Element对象所以它们的操作方式都是相同的
- }
- //但是有些情况xml比较复杂,规范不统一,某个节点不存在直接java.lang.NullPointerException,所以获取到element对象之后要先判断一下是否为空
- rootElement.setName("root");//支持修改节点名称
- System.out.println("根节点修改之后的名称:" + rootElement.getName());
- rootElement.setText("text"); //同样修改标签内的文本也一样
- System.out.println("根节点修改之后的文本:" + rootElement.getText());
- } catch (Exception e) {
- e.printStackTrace();
- }
- }
- public static void main(String[] args) {
- Dom4jParseXmlDemo demo = new Dom4jParseXmlDemo();
- demo.parseXml01();
- }
- }
- <?xml version="1.0" encoding="UTF-8"?>
- <modules id="123">
- <module> 这个是module标签的文本信息</module>
- </modules>
由此以知:
<1>dom4j读取xml文件方式有很多样;
<2>取出element对象的文本和标签名称都非常简单;
<3>并且修改元素的文本和标签名称都非常方便,但是不会写入到磁盘xml文件中。
上面只是简单的获取了xml的根目录的元素,接下来使用Iterator 迭代器循环document文档对象。
具体代码如下:
- public void parseXml02(){
- try{
- //将src下面的xml转换为输入流
- InputStream inputStream = this.getClass().getResourceAsStream("/module02.xml");
- //创建SAXReader读取器,专门用于读取xml
- SAXReader saxReader = new SAXReader();
- //根据saxReader的read重写方法可知,既可以通过inputStream输入流来读取,也可以通过file对象来读取
- Document document = saxReader.read(inputStream);
- Element rootElement = document.getRootElement();
- Iterator<Element> modulesIterator = rootElement.elements("module").iterator();
- //rootElement.element("name");获取某一个子元素
- //rootElement.elements("name");获取根节点下子元素moudule节点的集合,返回List集合类型
- //rootElement.elements("module").iterator();把返回的list集合里面每一个元素迭代子节点,全部返回到一个Iterator集合中
- while(modulesIterator.hasNext()){
- Element moduleElement = modulesIterator.next();
- Element nameElement = moduleElement.element("name");
- System.out.println(nameElement.getName() + ":" + nameElement.getText());
- Element valueElement = moduleElement.element("value");
- System.out.println(valueElement.getName() + ":" + valueElement.getText());
- Element descriptElement = moduleElement.element("descript");
- System.out.println(descriptElement.getName() + ":" + descriptElement.getText());
- }
- } catch (Exception e) {
- e.printStackTrace();
- }
- }
- <?xml version="1.0" encoding="UTF-8"?>
- <modules id="123">
- <module>
- <name>oa</name>
- <value>系统基本配置</value>
- <descript>对系统的基本配置根目录</descript>
- </module>
- </modules>
由此以知:
<1>dom4j迭代xml子元素非常的效率和便捷;
但是上面只是简单的迭代了xml的子节点元素,但是如果xml规则比较复杂,比如接下来要测试的module03.xml,具体如下:
- <?xml version="1.0" encoding="UTF-8"?>
- <modules id="123">
- <module>这个是module标签的文本信息</module>
- <module id="">
- <name>oa</name>
- <value>系统基本配置</value>
- <descript>对系统的基本配置根目录</descript>
- <module>这个是子module标签的文本信息</module>
- </module>
- <module>
- <name>管理配置</name>
- <value>none</value>
- <descript>管理配置的说明</descript>
- <module id="106">
- <name>系统管理</name>
- <value>0</value>
- <descript>Config</descript>
- <module id="107">
- <name>部门编号</name>
- <value>20394</value>
- <descript>编号</descript>
- </module>
- </module>
- </module>
- </modules>
java.lang.NullPointerException
所以这个时候需要小心使用了,每次都不能把元素直接放进去迭代。具体实现代码如下:
- public void parseXml03(){
- try{
- //将src下面的xml转换为输入流
- InputStream inputStream = this.getClass().getResourceAsStream("/module03.xml");
- //创建SAXReader读取器,专门用于读取xml
- SAXReader saxReader = new SAXReader();
- //根据saxReader的read重写方法可知,既可以通过inputStream输入流来读取,也可以通过file对象来读取
- Document document = saxReader.read(inputStream);
- Element rootElement = document.getRootElement();
- if(rootElement.elements("module") != null ){
- //因为第一个module标签只有内容没有子节点,直接.iterator()就java.lang.NullPointerException了, 所以需要分开实现
- List<Element> elementList = rootElement.elements("module");
- for (Element element : elementList) {
- if(!element.getTextTrim().equals("")){
- System.out.println("【1】" + element.getTextTrim());
- }else{
- Element nameElement = element.element("name");
- System.out.println(" 【2】" + nameElement.getName() + ":" + nameElement.getText());
- Element valueElement = element.element("value");
- System.out.println(" 【2】" + valueElement.getName() + ":" + valueElement.getText());
- Element descriptElement = element.element("descript");
- System.out.println(" 【2】" + descriptElement.getName() + ":" + descriptElement.getText());
- List<Element> subElementList = element.elements("module");
- for (Element subElement : subElementList) {
- if(!subElement.getTextTrim().equals("")){
- System.out.println(" 【3】" + subElement.getTextTrim());
- }else{
- Element subnameElement = subElement.element("name");
- System.out.println(" 【3】" + subnameElement.getName() + ":" + subnameElement.getText());
- Element subvalueElement = subElement.element("value");
- System.out.println(" 【3】" + subvalueElement.getName() + ":" + subvalueElement.getText());
- Element subdescriptElement = subElement.element("descript");
- System.out.println(" 【3】" + subdescriptElement.getName() + ":" + subdescriptElement.getText());
- }
- }
- }
- }
- }
- } catch (Exception e) {
- e.printStackTrace();
- }
- }
好了,这下可以解决迭代文档出现空引用的情况了。
另外代码其实可以重构一下,因为循环里面取出子元素的操作都是重复的,可以利用递归改善,但是可读性会变差一点。
如果有些时候需要获取xml中所有的文本信息,又或者别人传递的xml格式不规范,比如标签内名称大小写,虽然xml不区分大小写,但是必须成对出现,所以为了避免这种情况,索性可以将全部的标签名称换为大写,具体代码如下:
- public static void main(String[] args) {
- String str = "<?xml version=\"1.0\" encoding=\"UTF-8\"?><modules id=\"123\"><module> 这个是module标签的文本信息<name>oa</name><value>系统基本配置</value><descript>对系统的基本配置根目录</descript></module></modules>";
- System.out.println(str.replaceAll("<[^<]*>", "_"));
- Pattern pattern = Pattern.compile("<[^<]*>");
- Matcher matcher = pattern.matcher(str);
- while(matcher.find()){
- str = str.replaceAll(matcher.group(0), matcher.group(0).toUpperCase());
- }
- System.out.println(str);
- }
2、生成xml文档
dom4j能够解析xml,同样肯定能生成xml,而且使用起来更加简单方便。
实现思路:
<1>DocumentHelper提供了创建Document对象的方法;
<2>操作这个Document对象,添加节点以及节点下的文本、名称和属性值;
<3>然后利用XMLWriter写入器把封装的document对象写入到磁盘中;
具体代码如下:
- import java.io.FileWriter;
- import java.io.IOException;
- import java.io.Writer;
- import org.dom4j.Document;
- import org.dom4j.DocumentHelper;
- import org.dom4j.Element;
- import org.dom4j.io.XMLWriter;
- /**
- * 使用dom4j生成xml文档
- * @author Administrator
- *
- */
- public class Dom4jBuildXmlDemo {
- public void build01(){
- try {
- //DocumentHelper提供了创建Document对象的方法
- Document document = DocumentHelper.createDocument();
- //添加节点信息
- Element rootElement = document.addElement("modules");
- //这里可以继续添加子节点,也可以指定内容
- rootElement.setText("这个是module标签的文本信息");
- Element element = rootElement.addElement("module");
- Element nameElement = element.addElement("name");
- Element valueElement = element.addElement("value");
- Element descriptionElement = element.addElement("description");
- nameElement.setText("名称");
- nameElement.addAttribute("language", "java");//为节点添加属性值
- valueElement.setText("值");
- valueElement.addAttribute("language", "c#");
- descriptionElement.setText("描述");
- descriptionElement.addAttribute("language", "sql server");
- System.out.println(document.asXML()); //将document文档对象直接转换成字符串输出
- Writer fileWriter = new FileWriter("c:\\module.xml");
- //dom4j提供了专门写入文件的对象XMLWriter
- XMLWriter xmlWriter = new XMLWriter(fileWriter);
- xmlWriter.write(document);
- xmlWriter.flush();
- xmlWriter.close();
- System.out.println("xml文档添加成功!");
- } catch (IOException e) {
- e.printStackTrace();
- }
- }
- public static void main(String[] args) {
- Dom4jBuildXmlDemo demo = new Dom4jBuildXmlDemo();
- demo.build01();
- }
- }
运行代码效果如下:
然后去c盘下面查看是否创建成功,结果发现在xml文件中的内容与控制台输出的内容一样。
另外上面生成xml并没有指定编码格式,但是还是显示了UTF-8,说明这个是默认的编码格式,如果想重新指定可以在写入到磁盘之前加上document.setXMLEncoding("GBK");就好了。
相关推荐
- 初学者遇到java.lang.NoClassDefFoundError: org/dom4j/DocumentException解决方法
- 处理java环境配置后出出现 javac不是内部或外部命令
- java.sql.SQLException: GC overhead limit exceeded处理百万数据出现的异常
- Java编程GitHub之处理冲突
- Java可以这样处理json,你还不知道吗?
- java中xml转json
- (Java)常用数据类型处理类实验
- dea中出现MyBatis的 java.io.IOException: Could not find resource cn.wyu.dao.UserDao.xml异常处理。
- WebService学习(九)——cxf处理java bean及List、Map类型
- 使用idea将java程序打包成批处理文件
- java环境的配置和求最大子数组
- 求环形数组的最大子数组和