XML解析和保存

XML解析

1.解析思路
DOM解析Document Object Model 文件对象解析
把整个xml文件看作一个Document对象每一个节点看作一个Element,节点中有Attribute,或者当前节点中保存在Text文本内容
DOM是将整个xml文件读取都计算机内存中,可以进行CRUD操作
缺点:占用了大量的内存空间
适用环境:
服务器对于xml文件的解析过程
SAX解析
逐行读取,给予一定的时间操作
读取一行内容,释放上一行内容,可以有效地节约内存空间
缺点:
不能对xml文件进行增删改查
使用环境:
手机读取解析XML文件是采用的方式
DOM解析过程可以大致看作:

Document对象获取整个XML文件,通过获得根节点对象设置属性,然后进而获得其他的节点对象获取内容

在这里插入图片描述

XML文件解析工具

  1. JAXP: SUN提供的一个基本的解析器,支持DOM和SAX两种解析方式,但是操作很繁琐,不便于程序员开发。
  2. Dom4j: DOM For Java 一款非常优秀的解析器 Spring,SpringMVC… 框架中集成的XML解析器
  3. Jsoup: 基于Java完成的对于HTML解析的工具,因为HTML和XML文件都是标记语言。 给Jsoup一个URL,页面地址. Java的小爬虫,API很多很方便
  4. PULL: Android手机上集成的XML解析工具,SAX方式解析

Dom4j使用入门
1.导包
目前使用的是第三方工具,不是原生jdk
导入第三方jar包
2. 设置IDEA
一些方法
SAXReader reader = new SAXReader();
解析xml的核心类
Document document = reader.read(file);
Document document = new SAXReader().read(new File("./xml/User.xml"));
Document对象中可以使用
Element getRootElement();
获取当前xml文件的根节点
Element对象中可以使用方法
List elements();
Attribute attribute(String name);
根据属性名获取对应的属性对象Attribute
Attribute节点中可以使用String getValue()来获取对应的节点数据
String getName();
获取当前节点的名字
String getText();
获取当前节点对应的文本数据

package com.qfedu.d_xmlparse;

import jdk.nashorn.internal.ir.CallNode;
import org.dom4j.Document;
import org.dom4j.DocumentException;
import org.dom4j.Element;
import org.dom4j.io.SAXReader;

import java.io.File;
import java.util.List;

/**
 * @description Dom4j方法演示
 * @author Anonymous
 * @data 2020/3/6 15:07
 */
public class Demo1 {
    public static void main(String[] args) throws DocumentException {

        // 当前XML文件的Document对象
        Document document = new SAXReader().read(new File("./xml/User.xml"));

        // 获取根节点 Root Element
        Element rootElement = document.getRootElement();

        // 获取Root Element下子节点
        List<Element> elements = rootElement.elements();
        // 可以获取当前节点下的所有子节点
        // System.out.println(elements);
        Element teacher = rootElement.element("teacher");

        // 获取属性节点指定属性值
        System.out.println(teacher.attribute("id").getValue());
        List<Element> elements1 = teacher.elements();
        for (Element element : elements1) {
            // 获取节点的名字
            System.out.println(element.getName() + ":" + element.getText());
        }
    }
}
package com.qfedu.d_xmlparse;

import org.dom4j.Document;
import org.dom4j.DocumentException;
import org.dom4j.Element;
import org.dom4j.Node;
import org.dom4j.io.SAXReader;

import java.io.File;
import java.util.List;

/**
 * @description Xpath语法解析 XML文件
 * @author Anonymous
 * @date 2020/3/6 15:55
 */
public class Demo2 {
    public static void main(String[] args) throws DocumentException {
        // 当前XML文件的Document对象
        Document document = new SAXReader().read(new File("./xml/User.xml"));

        // 获取根节点 Root Element
        Element rootElement = document.getRootElement();

        // 无视目录结构,无视当前解析状态,直接获取所有的user节点
        List list = rootElement.selectNodes("//user");
        for (Object o : list) {
            Element element = (Element) o;

            int id = Integer.parseInt(element.attributeValue("id"));
            String name = element.element("name").getText();
            int age = Integer.parseInt(element.element("age").getText());
            String gender = element.element("gender").getText();

            System.out.println("User[id=" + id + ", name=" + name
                    + ", age=" + age + ", gender=" + gender + "]");
        }

        System.out.println("--------------------------------------");
        System.out.println();

        // //user[index] 不考虑路径关系,获取当前XML文件中指定下标的user节点
        Node node = rootElement.selectSingleNode("//user[1]");
        System.out.println(node);

        System.out.println("--------------------------------------");
        System.out.println();

        // //user[@id=9] 不考虑路径关系,获取当前XML文件中指定属性为id,值为9的user节点
        // @之后是属性 @id 这里需要匹配的是属性
        Node node1 = rootElement.selectSingleNode("//user[@id=9]");
        System.out.println(node1);

        System.out.println("--------------------------------------");
        System.out.println();

        // //user[gender='male'] 不考虑路径关系,获取当前XML文件中指定子节点为gender,对应子节点文本数据为male
        // 的user所有节点
        List list1 = rootElement.selectNodes("//user[gender='male']");
        for (Object o : list1) {
            System.out.println(o);
        }

        System.out.println("--------------------------------------");
        System.out.println();

        List list2 = rootElement.selectNodes("//user[age<20]");
        for (Object o : list2) {
            System.out.println(o);
        }
    }
}
package com.qfedu.d_xmlparse;

import org.dom4j.Document;
import org.dom4j.DocumentException;
import org.dom4j.Element;
import org.dom4j.Node;
import org.dom4j.io.SAXReader;

import java.io.File;
import java.util.List;

/**
 * @description Xpath语法解析 XML文件
 * @author Anonymous
 * @date 2020/3/6 15:55
 */
public class Demo2 {
    public static void main(String[] args) throws DocumentException {
        // 当前XML文件的Document对象
        Document document = new SAXReader().read(new File("./xml/User.xml"));

        // 获取根节点 Root Element
        Element rootElement = document.getRootElement();

        // 无视目录结构,无视当前解析状态,直接获取所有的user节点
        List list = rootElement.selectNodes("//user");
        for (Object o : list) {
            Element element = (Element) o;

            int id = Integer.parseInt(element.attributeValue("id"));
            String name = element.element("name").getText();
            int age = Integer.parseInt(element.element("age").getText());
            String gender = element.element("gender").getText();

            System.out.println("User[id=" + id + ", name=" + name
                    + ", age=" + age + ", gender=" + gender + "]");
        }

        System.out.println("--------------------------------------");
        System.out.println();

        // //user[index] 不考虑路径关系,获取当前XML文件中指定下标的user节点
        Node node = rootElement.selectSingleNode("//user[1]");
        System.out.println(node);

        System.out.println("--------------------------------------");
        System.out.println();

        // //user[@id=9] 不考虑路径关系,获取当前XML文件中指定属性为id,值为9的user节点
        // @之后是属性 @id 这里需要匹配的是属性
        Node node1 = rootElement.selectSingleNode("//user[@id=9]");
        System.out.println(node1);

        System.out.println("--------------------------------------");
        System.out.println();

        // //user[gender='male'] 不考虑路径关系,获取当前XML文件中指定子节点为gender,对应子节点文本数据为male
        // 的user所有节点
        List list1 = rootElement.selectNodes("//user[gender='male']");
        for (Object o : list1) {
            System.out.println(o);
        }

        System.out.println("--------------------------------------");
        System.out.println();

        List list2 = rootElement.selectNodes("//user[age<20]");
        for (Object o : list2) {
            System.out.println(o);
        }
    }
}

XML文件保存

流程:
1. 创建Document对象
2. 通过Document对象来添加元素
addElment();
addAttribute();

package com.qfedu.e_xmlwrite;

import org.dom4j.Document;
import org.dom4j.DocumentHelper;
import org.dom4j.Element;
import org.dom4j.io.OutputFormat;
import org.dom4j.io.XMLWriter;

import java.io.FileWriter;
import java.io.IOException;

/**
 * @description: 创建XML文件
 * @author: Anonymous
 * @time: 2020/3/6 16:29
 *
 * <students>
 *     <student id="qf1">
 *         <name>骚磊</name>
 *         <age>fdasfdsaf</age>
 *         <sex>male</sex>
 *     </student>
 *     <student id="qf2">
 *         <name>骚磊</name>
 *         <age>16</age>
 *         <sex>male</sex>
 *     </student>
 * </students>
 */
public class Demo1 {
    public static void main(String[] args) throws IOException {
        // 1. 创建了一个XML文件对应的Document对象
        Document document = DocumentHelper.createDocument();

        // 2. 放入到XML文件对应的Document对象中
        Element root = document.addElement("students");

        // 3. 在根节点之下添加了一个student节点,同时设置了属性
        Element element = root.addElement("student").addAttribute("id", "qf1");
        // 根节点下添加了对应的子节点,以及对应的文本
        element.addElement("name").addText("骚磊");
        element.addElement("age").addText("16");
        element.addElement("sex").addText("male");

        // 字符输出流
        FileWriter fileWriter = new FileWriter("./xml/student.xml");
        document.write(fileWriter);
        // 4. 关闭资源
        fileWriter.close();
    }
}

反射+XML文件保存读取操作

package f_xmlsave;

import org.dom4j.Document;
import org.dom4j.DocumentException;
import org.dom4j.DocumentHelper;
import org.dom4j.Element;
import org.dom4j.io.SAXReader;

import java.io.File;
import java.io.FileWriter;
import java.io.IOException;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.List;

/**
 * @description:
 * @author: Anonymous
 * @time: 2020/3/6 16:46
 */
public class MainProject {
    public static void main(String[] args)
            throws IllegalAccessException, DocumentException, NoSuchMethodException, InvocationTargetException, IOException {
        ArrayList<Student> list = new ArrayList<>();

        readDataFromXML(list);

        for (Student student : list) {
            System.out.println(student);
        }
    }

    public static void saveDateToXML(ArrayList<Student> list) throws IllegalAccessException, IOException {
        // 创建XML文件对应Document对象
        Document document = DocumentHelper.createDocument();

        // 明确根节点
        Element root = document.addElement("students");

        // 获取所有的成员变量Field对象
        Field[] declaredFields = Student.class.getDeclaredFields();

        // 循环遍历Student ArrayList集合
        for (Student student : list) {
            // 每一个Student对象都要对应一个Student节点
            Element element = root.addElement("student");

            // 遍历所有的Field成员变量
            for (Field declaredField : declaredFields) {
                declaredField.setAccessible(true);

                // id存储到Student节点中的属性中
                if ("id".equals(declaredField.getName())) {
                    // 所有的数据都是在String类型处理
                    element.addAttribute("id", declaredField.get(student) + "");
                } else {
                    // declaredField.getName() 成员变量名字 declaredField.get(student) 对应数据
                    element.addElement(declaredField.getName()).addText(declaredField.get(student) + "");
                }
            }
        }

        // 字符流对象+Document对象的write方法写入XML信息到文件中
        FileWriter fileWriter = new FileWriter("./xml/student.xml");
        document.write(fileWriter);
        fileWriter.close();
    }

    public static void readDataFromXML(ArrayList<Student> list)
            throws DocumentException, NoSuchMethodException, InvocationTargetException, IllegalAccessException {
        Document document = new SAXReader().read(new File("./xml/student.xml"));

        // 找出当前XML文件中的所有student节点
        List list1 = document.selectNodes("//student");

        // 得到当前Student类内的所有成员变量对象,注意使用暴力反射
        Field[] declaredFields = Student.class.getDeclaredFields();

        /*
        String字符串问题
            name String getText
            gender String getText

            id Integer Attribute ==> Integer
            age Integer getText ==> Integer
         */

        // 遍历所有的Student节点
        for (Object o : list1) {
            // Student节点对象
            Element element = (Element) o;
            Student student = new Student();

            // 成员变量Field数组遍历
            for (Field declaredField : declaredFields) {
                // 给予暴力反射操作成员变量权限
                declaredField.setAccessible(true);

                // 获取当前成员变量的数据类型
                Class<?> type = declaredField.getType();

                // 如果数据类型是String类型
                if (type.equals(String.class)) {
                    // String
                    declaredField.set(student, element.element(declaredField.getName()).getText());
                } else if (type.equals(Integer.class)) {
                    // Integer类型
                    // 获取Integer类型中的valueOf方法
                    Method valueOf = type.getMethod("valueOf", String.class);

                    if ("id".equals(declaredField.getName())) {
                        /*
                        id是在student节点属性中,从属性中获取对应是数据,使用valueOf方法转换成对应的Integer类型
                         */
                        declaredField.set(student, valueOf.invoke(student, element.attributeValue("id")));
                    } else {
                        /*
                        非ID数据,从Student指定名字的子节点下获取,指定名字和成员变量名字一直,同样需要转换一下
                         */
                        declaredField.set(student,valueOf.invoke(student, element.element(declaredField.getName()).getText()));
                    }
                }
            }

            list.add(student);
        }

    }
}

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值