Java SE 学习笔记(十九)—— XML、设计模式

目录

  • 1 XML
    • 1.1 XML 概述
    • 1.2 XML 语法规则
    • 1.3 XML 文档约束(了解)
      • 1.3.1 DTD 约束
      • 1.3.2 schema 约束
  • 2 XML 解析
    • 2.1 XML 解析概述
    • 2.2 Dom4J 解析 XML 文件
    • 2.3 XML 解析案例
  • 3 XML 检索
  • 4 设计模式
    • 4.1 工厂模式
    • 4.2 装饰模式

1 XML


在有些业务场景下,存储数据或者传输数据给别人的时候需要满足一定的规范进行组织

1.1 XML 概述


XML 的全称为(EXtensible Markup Language),是一种 可扩展标记语言,是一种数据表示格式,可以用于自定义数据格式,可以描述非常复杂的数据结构,常用于传输和存储数据。

例如:

<?xml version="1.0" encoding="UTF-8"?>
<data><sender> 张三 </sender><receiver> 李四 </receiver><src><addr> 北京 </addr><date>2022-11-11 11:11:11</date></src><current> 武汉 </current><dest> 广州 </dest>
</data>

XML 的几个特点和使用场景

  • 一是纯文本,默认使用 UTF-8 编码;二是可嵌套;
  • 如果把 XML 内容存为文件,那么它就是一个 XML 文件。
  • XML 内容经常被 当成消息进行网络传输,或者 作为配置文件用于存储系统的信息

1.2 XML 语法规则


  1. XML 的创建

就是创建一个 XML 类型的文件,要求文件的后缀名必须使用 xml,如:helloworld.xml

  1. XML 文件的文档声明

XML 文件的文档声明必须在第一行

<?xml version="1.0" encoding="UTF-8" ?>

version : XML 默认的版本号码、该属性是必须存在的
encoding :本 XML 文件的编码

  1. XML 的标签 ( 元素 ) 规则
  • 标签由一对尖括号和合法标识符组成:<name></name>,必须存在一个根标签,有且只能有一个
  • 标签必须成对出现,有开始,有结束 : <name></name>
  • 特殊的标签可以不成对,但是必须有结束标记,如 :<br/>
  • 标签中可以定义属性,属性和标签名空格隔开 ,属性值必须用引号引起来 <student id = “1”></name>
  • 标签需要正确的嵌套
  • XML 文件中可以定义注释信息: <!-- 注释内容 -->
  • XML 文件中可以存在以下特殊字符
    • &lt;< 小于
    • &gt;> 大于
    • &amp;& 和号
    • &apos;' 单引号
    • &quot;" 引号
  • XML 文件可以存在 CDATA 区 (IDEA中输入CD再回车)

示例代码

<?xml version="1.0" encoding="UTF-8" ?>
<student><msg>age &gt; 18 &amp;&amp; age &lt;24<![CDATA[age>18 && age<24]]></msg>
</student>

1.3 XML 文档约束(了解)


问题:由于 XML 文件可以自定义标签,导致 XML 文件可以随意定义,程序在解析的时候可能出现问题。

文档约束是用来限定 XML 文件中的标签以及属性应该怎么写,以此强制约束程序员必须按照文档约束的规定来编写 XML 文件中

文档约束可以分为:DTD和 schema 两种

1.3.1 DTD 约束


需求:利用 DTD 文档约束,约束一个 XML 文件的编写。

分析:

  • 编写 DTD 约束文档,后缀必须是 .dtd
  • 在需要编写的 XML 文件中导入该 DTD 约束文档
  • 按照约束的规定编写 XML 文件的内容

示例代码

在这里插入图片描述

XML 的文档约束 -DTD 的作用和问题?

  • 可以约束 XML 文件的编写。
  • 不能约束具体的数据类型。

1.3.2 schema 约束


schema 可以约束具体的数据类型,约束能力上更强大,但由于其本身也是一个 XML 文件,也受到其他约束文件的要求,所以编写地更严谨

需求:利用 schema 文档约束,约束一个 XML 文件的编写。

分析:

  • 编写 schema 约束文档,后缀必须是 .xsd
  • 在需要编写的 XML 文件中导入该 schema 约束文档
  • 按照约束内容编写 XML 文件的标签。

示例代码

在这里插入图片描述

2 XML 解析

2.1 XML 解析概述


XML 的数据的作用是什么,最终需要怎么处理?

  • 存储数据、做配置信息、进行数据传输。
  • 最终需要被程序进行读取,解析里面的信息。

主要有两种解析方式: SAX 解析、DOM 解析

Dom 常见的解析工具

在这里插入图片描述

DOM 解析解析文档对象模型

在这里插入图片描述

Document 对象:整个 xml 文档
Node 对象

  • Element 对象:标签
  • Attribute 对象:属性
  • Text 对象:文本内容

2.2 Dom4J 解析 XML 文件


需求:使用 Dom4J 把一个 XML 文件的数据进行解析

分析:

  • 下载 Dom4j 框架,官网下载。
  • 在项目中创建一个文件夹: lib
  • 将 dom4j-2.1.1.jar 文件复制到 lib 文件夹
  • 在 jar 文件上点右键,选择 Add as Library -> 点击 OK
  • 在类中导包使用

Dom4j 解析 XML- 得到 Document 对象

SAXReader类

在这里插入图片描述

Document 类

在这里插入图片描述

Dom4j 解析 XML 的元素、属性、文本

在这里插入图片描述

示例代码

XML文件

<?xml version="1.0" encoding="UTF-8"?>
<contactList><contact id="1" vip="true">哈哈哈<name>   潘金莲  </name><gender></gender><email>panpan@it.cn</email></contact><contact id="2" vip="false"><name>武松</name><gender></gender><email>wusong@it.cn</email></contact><contact id="3" vip="false"><name>武大狼</name><gender></gender><email>wuda@it.cn</email></contact><user></user>
</contactList>

解析XML文件

import org.dom4j.Attribute;
import org.dom4j.Document;
import org.dom4j.Element;
import org.dom4j.io.SAXReader;
import org.junit.Test;
import java.io.InputStream;public class Dom4jDemo {@Testpublic void parseXML() throws Exception{// 1. 创建一个Dom4j解析器对象,代表了整个Dom4j框架SAXReader saxReader = new SAXReader();// 2. 把XML文件加载到内存中称为一个Document对象
//        Document document = saxReader.read("xml\\src\\Contacts.xml");
//        Document document = saxReader.read(new File("xml\\src\\Contacts.xml"));
//        Document document = saxReader.read(new FileInputStream("xml\\src\\Contacts.xml"));// getResourceAsStream 中的/是直接去src下寻找文件InputStream is = Dom4jDemo.class.getResourceAsStream("/Contacts.xml");Document document = saxReader.read(is);// 3. 获取根元素对象Element rootElement = document.getRootElement();System.out.println(rootElement.getName()); // contactList// 4. 获取子元素// 4.1 获取根元素下所有的子元素(一级)
//        List<Element> elements = rootElement.elements();
//        for (Element element : elements) {
//            System.out.println(element.getName());
//        }// 4.2 获取某个子元素(若同名,默认提取第一个子元素对象)Element contact = rootElement.element("contact");System.out.println(contact.getName()); // contact// 4.3 获取当前元素的文本System.out.println(contact.getText());// 去掉前后空格System.out.println(contact.getTextTrim()); // 哈哈哈// 4.4 获取子元素文本System.out.println(contact.elementText("name"));// 去掉前后空格System.out.println(contact.elementTextTrim("name")); // 潘金莲// 4.5 根据元素获取属性值Attribute id = contact.attribute("id");// 打印属性名和属性值System.out.println(id.getName()+":"+id.getValue());// 4.6 直接获取属性值System.out.println(contact.attributeValue("id")); // 1System.out.println(contact.attributeValue("vip")); // true}
}

2.3 XML 解析案例


需求 : 利用 Dom4J 的知识,将 Contact.xml 文件中的联系人数据封装成 List 集合,其中每个元素是实体类 Contact 。打印输出 List 中的每个元素。

案例用到的 XML 文件见上文

创建 Contact 类

public class Contact {private String name;private int id;private boolean vip;private char gender;private String email;public Contact() {}public Contact(String name, int id, boolean vip, char gender, String email) {this.name = name;this.id = id;this.vip = vip;this.gender = gender;this.email = email;}public String getName() {return name;}public void setName(String name) {this.name = name;}public int getId() {return id;}public void setId(int id) {this.id = id;}public boolean isVip() {return vip;}public void setVip(boolean vip) {this.vip = vip;}public char getGender() {return gender;}public void setGender(char gender) {this.gender = gender;}public String getEmail() {return email;}public void setEmail(String email) {this.email = email;}@Overridepublic String toString() {return "Contact{" +"name='" + name + '\'' +", id=" + id +", vip=" + vip +", gender=" + gender +", email='" + email + '\'' +'}';}
}

解析XML

import org.dom4j.Document;
import org.dom4j.Element;
import org.dom4j.io.SAXReader;
import org.junit.Test;import java.lang.invoke.StringConcatException;
import java.util.ArrayList;
import java.util.List;public class TestDemo {@Testpublic void parseToList() throws Exception {// 1. 创建saxReader对象SAXReader saxReader = new SAXReader();// 2. 加载XML文件成为Document对象Document document = saxReader.read(TestDemo.class.getResourceAsStream("/Contacts.xml"));// 3. 先获取根元素Element rootElement = document.getRootElement();// 4. 获取所有的contact子元素List<Element> contactEles = rootElement.elements("contact");// 5. 准备一个ArrayList集合封装联系人信息ArrayList<Contact> contactsList = new ArrayList<>();// 6. 遍历每一个contact子元素for (Element contactEle : contactEles) {// 7. 每一个contact子元素都是一个联系人对象Contact contact = new Contact();// 8. 提取信息并给对象赋值contact.setId(Integer.valueOf(contactEle.attributeValue("id")));contact.setVip(Boolean.valueOf(contactEle.attributeValue("vip")));contact.setName(contactEle.elementTextTrim("name"));contact.setGender(contactEle.elementTextTrim("gender").charAt(0));contact.setEmail(contactEle.elementTextTrim("email"));// 9.把联系人对象放入List集合contactsList.add(contact);}// 10. 遍历集合for (Contact contact : contactsList) {System.out.println(contact);}}
}

输出结果

在这里插入图片描述

总结

  • Dom4J 解析 XML 文件的核心思想
    • 得到文档对象 Document ,从中获取元素对象和内容。
  • Dom4J 的解析后的数据形式。
    • 通常数据会封装成 Java 的对象,如单个对象,或者集合对象形
      式。

3 XML 检索


如果需要从 XML 文件中检索需要的某个信息(如 name )怎么解决?

  • Dom4j 需要进行文件的全部解析,然后再寻找数据。
  • Xpath 技术更加适合做信息检索。

XPath 在解析 XML 文档方面提供了一独树一帜的路径思想,更加优雅,高效。XPath 使用路径表达式来定位 XML 文档中的元素节点或属性节点。

使用 Xpath 检索出 XML 文件

需求:使用 Dom4J 把一个 XML 文件的数据进行解析

分析:

  • 导入 jar 包 (dom4j 和 jaxen-1.1.2.jar) , Xpath 技术依赖 Dom4j 技术
  • 通过 dom4j 的 SAXReader 获取 Document 对象
  • 利用 XPath 提供的 API, 结合 XPath 的语法完成选取 XML 文档元素节点进行解析操作。

Document 中与 Xpath 相关的 API

在这里插入图片描述

Xpath 的四大检索方案

  1. 绝对路径:采用绝对路径获取从根节点开始逐层的查找 /contactList/contact/name 节点列表并打印信息

在这里插入图片描述

  1. 相对路径:先得到根节点 contactList,再采用相对路径获取下一级 contact 节点的 name 子节点并打印信息

在这里插入图片描述

  1. 全文检索:直接全文搜索所有的 name 元素并打印

在这里插入图片描述

  1. 属性查找:在全文中搜索属性,或者带属性的元素

在这里插入图片描述

示例代码

XML文件

<?xml version="1.0" encoding="UTF-8"?>
<contactList><contact id="1" vip="true"><name>   潘金莲  </name><gender></gender><email>panpan@itcast.cn</email></contact><contact id="2" vip="false"><name>武松</name><gender></gender><email>wusong@itcast.cn</email></contact><contact id="3" vip="false"><name>武大狼</name><gender></gender><email>wuda@itcast.cn</email></contact><user><contact><info><name id="666">西门庆</name></info></contact></user>
</contactList>

检索XML

import org.dom4j.Attribute;
import org.dom4j.Document;
import org.dom4j.Element;
import org.dom4j.Node;
import org.dom4j.io.SAXReader;
import org.junit.Test;import java.util.List;public class XPathDemo {// 1. 绝对路径@Testpublic void parse01() throws Exception {// 创建解析器对象SAXReader saxReader = new SAXReader();// 将XML文件加载成Document对象Document document = saxReader.read(XPathDemo.class.getResourceAsStream("/Contacts2.xml"));// 检索全部名称:name值List<Node> nodes = document.selectNodes("/contactList/contact/name");for (Node node : nodes) {Element nameEle = (Element) node;System.out.println(nameEle.getTextTrim());}}// 2. 相对路径@Testpublic void parse02() throws Exception {// 创建解析器对象SAXReader saxReader = new SAXReader();// 将XML文件加载成Document对象Document document = saxReader.read(XPathDemo.class.getResourceAsStream("/Contacts2.xml"));Element rootElement = document.getRootElement();// 检索全部名称:name值List<Node> nodes = rootElement.selectNodes("./contact/name");for (Node node : nodes) {Element nameEle = (Element) node;System.out.println(nameEle.getTextTrim());}}// 3. 全文搜素@Testpublic void parse03() throws Exception {// 创建解析器对象SAXReader saxReader = new SAXReader();// 将XML文件加载成Document对象Document document = saxReader.read(XPathDemo.class.getResourceAsStream("/Contacts2.xml"));// 检索数据
//        List<Node> nodes = document.selectNodes("//name");// 会找"西门庆"
//        List<Node> nodes = document.selectNodes("//contact/name");// 不会找"西门庆"List<Node> nodes = document.selectNodes("//contact//name");// 会找"西门庆"for (Node node : nodes) {Element nameEle = (Element) node;System.out.println(nameEle.getTextTrim());}}// 4. 属性查找@Testpublic void parse04() throws Exception {// 创建解析器对象SAXReader saxReader = new SAXReader();// 将XML文件加载成Document对象Document document = saxReader.read(XPathDemo.class.getResourceAsStream("/Contacts2.xml"));// 检索数据// 在全文中检索属性对象List<Node> nodes = document.selectNodes("//@id");for (Node node : nodes) {Attribute attr = (Attribute) node;System.out.println(attr.getName()+":"+attr.getValue());}// 在全文中检索包含该属性的元素对象(若有多个只返回第一个)// 查询name元素(包含id属性的)
//        Node node = document.selectSingleNode("//name[@id]");// 在全文中检索属性对象且属性值为该值的元素对象Node node = document.selectSingleNode("//name[@id=666]");Element ele = (Element) node;System.out.println(ele.getTextTrim());}
}

Element和继承Node

4 设计模式

4.1 工厂模式


之前我们创建类对象时,都是使用 new 对象的形式创建,在很多业务场景下也提供了不直接 new 的方式 。

工厂模式( Factory Pattern )是 Java 中最常用的设计模式之一, 这种类型的设计模式属于创建型模式,它提供了一种获取对象的方式。

工厂设计模式的作用:

  • 工厂的方法可以封装对象的创建细节,比如:为该对象进行加工和数据注入。
  • 可以实现类与类之间的解耦操作(核心思想)

4.2 装饰模式


装饰设计模式:创建一个新类,包装原始类,从而在新类中提升原来类的功能。

装饰设计模式的作用:

  • 装饰模式指的是在不改变原类的基础上 ,动态地扩展一个类的功能

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

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

相关文章

Go语言math标准库

# math包 参考资料 常用函数 函数功能备注NaN() float64返回一个IEEE 754“这不是一个数字”值IsNaN(f float64) (is bool)报告f是否表示一个NaN&#xff08;Not A Number&#xff09;值Inf(sign int) float64如果sign>0函数返回正无穷大&#xff0c;否则返回负无穷大IsI…

2022年上半年上午易错题(软件设计师考试)

1.以下关于冯诺依曼计算机的叙述中&#xff0c;不正确的是( )。 A.程序指令和数据都采用二进制表示 B.程序指令总是存储在主存中&#xff0c;而数据则存储在高速缓存中 C.程序的功能都由中央处理器(CPU)执行指令来实现 D.程序的执行过程由指令进行自动控制 程序指令和数据…

力扣第738题 单调递增的数字 c++ 暴力超时 贪心优化

题目 738. 单调递增的数字 中等 相关标签 贪心 数学 当且仅当每个相邻位数上的数字 x 和 y 满足 x < y 时&#xff0c;我们称这个整数是单调递增的。 给定一个整数 n &#xff0c;返回 小于或等于 n 的最大数字&#xff0c;且数字呈 单调递增 。 示例 1: 输入: n 1…

ZYNQ连载03-Vivado创建工程

ZYNQ连载03-Vivado创建工程 1. 硬件参数 名称参数主控xc7z020clg400-2DDRMT41J256M16RE-125 2. 创建工程 3. 串口配置 4. DDR配置 5. SD配置 6. ETH配置 7. USB配置 8. 导出硬件 Generate Output ProductsCreate HDL WrapperExport Hardware Platform 执行以上步骤后&#…

6G关键新兴技术-智能超表面(RIS)技术演进

一、产品定义及范围 根据欧盟5G公私联盟协会(5G Infrastructure Public-Private Partnership, 5GPP)定义&#xff0c;可重构智慧表面(Reconfigurable Intelligent Surface, RIS)技术是由能够任意塑造电磁波面的材料组成&#xff0c;几乎是被动(Passice)设备&#xff0c;可以适…

力扣每日一题81:搜索旋转排序数组||

题目描述&#xff1a; 已知存在一个按非降序排列的整数数组 nums &#xff0c;数组中的值不必互不相同。 在传递给函数之前&#xff0c;nums 在预先未知的某个下标 k&#xff08;0 < k < nums.length&#xff09;上进行了 旋转 &#xff0c;使数组变为 [nums[k], nums[…

leetCode 76. 最小覆盖子串 + 滑动窗口 + 哈希Hash

我的往期文章&#xff1a;此题的其他解法&#xff0c;感兴趣的话可以移步看一下&#xff1a; leetCode 76. 最小覆盖子串 滑动窗口 图解&#xff08;详细&#xff09;-CSDN博客https://blog.csdn.net/weixin_41987016/article/details/134042115?spm1001.2014.3001.5501 力…

【软件安装环境配置】VsCode安装和配置各种环境(保姆级)

一、VsCode 下载 1.官网下载 网站&#xff1a;Visual Studio Code - Code Editing. Redefined 打开网站 点击Download 根据操作系统&#xff08;macOS、Windows、Linux&#xff09;和版本下载 下载到本地 注意&#xff1a; 若下载很慢&#xff0c;或者下着下着就暂停了 可…

【设计模式】第15节:行为型模式之“职责链模式”

一、简介 职责链模式&#xff1a;将请求的发送和接收解耦&#xff0c;让多个接收对象都有机会处理这个请求。将这些接收对象串成一条链&#xff0c;并沿着这条链传递这个请求&#xff0c;直到链上的某个接收对象能够处理它为止。 二、优点 分离发送者和接受者易于扩展和维护…

【已解决】VSCode运行C#控制台乱码显示

问题描述 如上图所示&#xff0c;最近在学习C#突然发现我在运行Hello World的时候出现这样的乱码情况。 分析原因 主要是因为VS Code 是UTF-8的编码格式&#xff0c;而我们的PC是Unicode编码&#xff0c;所以我们需要对其进行一个统一即可解决问题。那么知道这个的问题那就开…

MATLAB算法实战应用案例精讲-【图像处理】机器视觉(基础篇)(六)

目录 前言 几个高频面试题目 工业相机与普通相机的差别 一、 工业相机与普通相机的区别

【Docker】如何查看之前docker run命令启动的参数

个人主页&#xff1a;金鳞踏雨 个人简介&#xff1a;大家好&#xff0c;我是金鳞&#xff0c;一个初出茅庐的Java小白 目前状况&#xff1a;22届普通本科毕业生&#xff0c;几经波折了&#xff0c;现在任职于一家国内大型知名日化公司&#xff0c;从事Java开发工作 我的博客&am…

QT之QDir的mkdir()和mkpath()区别

区别 QDir类的mkdir()和mkpath()方法都用于创建目录&#xff0c;但它们在处理多层目录的创建时有所不同。 mkdir()方法&#xff1a;它用于创建一个目录。如果要创建的目录的父目录不存在&#xff0c;则mkdir()方法会失败&#xff0c;不会创建任何目录。mkpath()方法&#xff…

H5游戏源码分享-密室逃脱小游戏(考验反应能力)

H5游戏源码分享-密室逃脱小游戏&#xff08;考验反应能力&#xff09; 预判安全位置&#xff0c;这个需要快速的反应能力 源码 <!DOCTYPE html> <html> <head> <meta http-equiv"Content-Type" content"text/html; charsetutf-8" /&…

Ubuntu 18.04 安装 Python3.8(安装包安装)

一、为什么用安装包&#xff0c;不用apt指令&#xff1f; 在使用apt指令安装时&#xff0c;莫名出现安装后的python3.8中没有pip&#xff0c;并且没有Math库&#xff01;&#xff01;&#xff01; ​​​​ 二、具体步骤 2.0准备环境 # 更新apt apt update # 安装相关包 apt…

1.spark standalone环境安装

概述 环境是spark 3.2.4 hadoop版本 3.2.4&#xff0c;所以官网下载的包为 spark-3.2.4-bin-hadoop3.2.tgz 在具体安装部署之前&#xff0c;需要先下载Spark的安装包&#xff0c;进到 spark的官网&#xff0c;点击download按钮 使用Spark的时候一般都是需要和Hadoop交互的&a…

2.7、图调度功能

introduction PPQ 的图调度功能 展示如何自由调度算子&#xff0c;并实现混合精度推理 code import torch import torchvision from ppq import * from ppq.api import *# ------------------------------------------------------------ # 在这个脚本中&#xff0c;我们将向…

10_4阻塞和非阻塞跟poll和等待队列

阻塞和非阻塞 广泛上的区别就是 应用程序如果非阻塞那读取不到数据就应该马上有返回值 阻塞的话就是在应用程序去read数据,但是设备驱动没有数据,就一直卡住,直到有数据再继续往下 补充阻塞知识,应用层大部分都是阻塞 如果要非阻塞 ,应用程序在打开设备节点的时候填写int fd …

辅助驾驶功能开发-功能规范篇(22)-4-L2级辅助驾驶方案功能规范

1.3.4 LDW系统功能定义 1.3.4.1 状态机 1.3.4.2 功能定义 1.3.4.2.1 信号需求列表 1.3.4.2.2 系统开启关闭 1)初始化 车辆上电后,车道偏离预警系统(LDW)进行初始化,控制器需要在上电后 220ms 内发出第一帧报文,并在 3s 内 完成内部自检,同时上电 3s 内不进行关联系统…

面经——java后端第一天

1.java和c区别是&#xff1f; 性能⾼&#xff1a;C 是⼀种编译型语⾔&#xff0c;可以⽣成⾼效的机器代码&#xff0c;所以在处理⼤量数据和对计算效率有要求的应⽤场景下具有优势&#xff1b; C 和 Java 是两种编程语言&#xff0c;它们之间有很多区别&#xff0c;主要包括以…