java jaxb 注解_一、JAXB注解的使用详解

article/2025/9/23 17:30:56

一、首先先上一段测试的代码下面的说明和测试都已这一段代码为主

package shiqingxue.cn.itcast.utils.xml;

import javax.xml.bind.annotation.XmlAccessType;

import javax.xml.bind.annotation.XmlAccessorType;

import javax.xml.bind.annotation.XmlElement;

import javax.xml.bind.annotation.XmlRootElement;

@XmlRootElement

public class Student {

private String name; // 姓名

private String sex; // 性别

private int number; // 学号

private String className; // 班级

public Student(){}

public Student(String string, String string2, int i, String string3) {

this.name = string;

this.sex = string2;

this.className = string3;

}

@XmlElement(name = "name")

public String getName() {

return name;

}

public void setName(String name) {

this.name = name;

}

@XmlElement(name = "sex")

public String getSex() {

return sex;

}

public void setSex(String sex) {

this.sex = sex;

}

@XmlElement(name = "number")

public int getNumber() {

return number;

}

public void setNumber(int number) {

this.number = number;

}

@XmlElement(name = "className")

public String getClassName() {

return className;

}

public void setClassName(String className) {

this.className = className;

}

}

/**

* 将XML内容转换成对象

*/

@SuppressWarnings("unchecked")

public static T unmarshal(String xml, Class clazz) throws JAXBException{

JAXBContext jaxbContext = JAXBContext.newInstance(clazz);

Unmarshaller unmarshaller = jaxbContext.createUnmarshaller();

return (T)unmarshaller.unmarshal(new StringReader(xml));

}

/**

* 将对象转换成XML

*/

public static String marshal(Object object, Class> clazz) throws JAXBException{

JAXBContext jaxbContext = JAXBContext.newInstance(clazz);

Marshaller marshaller = jaxbContext.createMarshaller();

marshaller.setProperty(Marshaller.JAXB_ENCODING, "UTF-8");

StringWriter writer = new StringWriter();

marshaller.marshal(object, writer);

return writer.toString();

}public static void main(String[] args) throws JAXBExceptionStudent st = new Student("张三","男",10001,"尖//这里试讲上面的两个方法放到了工具类在这里调String xml = XStreamUtil.marshal(st, Student.class)System.out.println(xml)}

测试结果:

eb335f573708c1afe9971bb22cf3fb4d.png二、可以看到上面控制台打印的结果,说明测试已经通过,下面具体将一些注解的用法:

(2.1)@XmlRootElement

类级别的注解,将类映射为xml全局元素,也就是根元素。就像spring配置文件中的beans。下面的例子中我将该注解用在了student类上,生成了根元素。常与@XmlType,@XmlAccessorType,@XmlAccessorOrder连用,看下面运行结果

d5c8b3d9836fe91fb5ae4464fc028699.png

6f87a8ab95f68700aaff9abc936f373b.png

明显可以看出生成的跟节点的名称已经变成了

(2.2)@XmlAccessorType:

包和类级别的注解,javaEE的API对该注解的解释是:控制字段是否被默认序列化。通俗来讲,就是决定哪些字段或哪些get/set方法对应的字段会被映射为xml元素,需要注意的是字段或get/set方法的访问权限(public/private)会影响字段是否被映射为xml元素,下面会详细讲解。

注解只有一个value属性,可取的值是一个名为XmlAccessType的枚举类型里的值,下面详细看一下这几个值分别有什么用:

(2.2.1)XmlAccessType.PROPERTY:

理解:下面是源码里面的一段描述,这一段的大致翻译是,《JAXB绑定类中的每个getter/setter对都将自动绑定到XML,除非由@link xmltinate注释,只有当字段被一些JAXB注释显式注释时,字段才绑定到XML》(注:这一段话个人认为需要格外注意前面的一段话每个getter/setter都会自动绑定到xml)。

d555c514813661f5e795a9b5fbbe5347.png

补充:

(1)当使用了该值,只要字段有对应的get/set方法对(注意是成对出现,只有其中一个不会发生映射),不需要使用@XmlElement注解,不论该方法的访问权限是什么(即使是private),jaxb就会将该字段映射成xml元素。不过最好加上@XmlElement注解,get/set方法任选一个即可,都加上会报错。

(2)若在一个字段有set/get方法对但又在字段上添加@XmlElement注解会报属性重复的错误

(3)若没有set/get方法对,则需要在字段上使用@XmlElement注解才可以映射为xml元素,否则不会发生映射

(4)若get/set方法上使用了@XmlTransient注解,但想要对应字段发生映射,需要在对应字段上添加@XmlElement注解,此时不会报错,并将该字段映射为xml元素。

(2.2.2)XmlAccessType.FIELD:

理解:下面是源码中的解释,这一段的大致翻译是,《jaxb绑定类中的每个非静态、非瞬态字段都将自动绑定到XML,除非使用@XmlTransient进行注释,只有当某些JAXB注释显式地对getter/setter对进行注释时,它们才会绑定到XML》(注:这段需要注意的是可以自动绑定类中的非静态、非瞬态字段,但不会自动绑定getter/setter方法,正好与XmlAccessType.PROPERTY相反)

8b075f78936058e51a1c417b9e6053f3.png

补充:

(1)每个非静态的字段(无论访问权限如何)都会被jaxb映射为xml元素,即使没有get/set方法对,即使没有使用@XmlElement元素,但最好加上该注解以表明该字段要被映射为xml元素

(2)虽然没有get/set方法对,也会发生映射,但加上get/set方法对也不会报错,因为我们经常会使用这两个方法。但注意,不能再在这两个方法上使用@XmlElement方法,否则会报属性重复的错误。

(3)若在字段上使用了@XmlTransient注解,但还想让该字段发生映射,需要在该字段对应的get/set方法上添加@XmlElement

(2.2.3)XmlAccessType.PUBLIC_MEMBER(该值为默认值):

注:如果不指定@XmlAccessorType的value值或者没有使用此注解,在转xml时会默认为value为XmlAccessType.PROPERTY,由下面的截图可以清晰的看出默认值是XmlAccessType.PROPERTY,

3c4eb3bb78e136d1cf0b36b4d3792708.png

理解:下面则是对此的描述,翻译大概是《每个公共getter/setter对和每个公共字段都将自动绑定到XML,除非使用@XmlTransient注释。只有在适当的JAXB注释显式地注释了字段或getter/setter对之后,才会将它们绑定到XML。》(注:在这里要格外注意XmlAccessType.PUBLIC_MEMBER (该值为默认值)这句话,同时他会把类的字段和getter/setter方法都会绑定到xml,初学者很容易忽略这点,采坑)

5b4dcfac615bd39bb24f1cf5c1510219.png

采坑实例:

如下图我们并没有使用@XmlAccessorType注解,当刚刚提到的,如果你在类的上面没有使用@XmlAccessorType注解,它的默认值会是XmlAccessType.PUBLIC_MEMBER,而XmlAccessType.PUBLIC_MEMBER的特性是会自动绑定类中的字段和getter/setter方法到XML,所以这样最终的结果会出现类的两个属性具有相同名称的错误,这一点需要注意

18fc234dcb62d2a7c915c07fdeabc3ce.png

2231e02c09c860a0835594aa341e4dc8.png

a68574260306fa28ba726f1cfc123550.png

补充:

(1)每个访问权限为public的字段,或者每个访问权限为public的get/set方法对,都会将字段映射为xml元素,即使不使用@XmlElement,但最好加上。不可同时存在public字段和对应的get/set方法对,不然会报属性重复的错误

(2)若使用@XmlElement注解,则实体类中不能存在get/set方法或者只能在get/set上使用,否则会报属性重复的错误

(3)若字段不为public,get/set方法为public并使用了@XmlTransient,需要在字段上添加@XmlElement才会发生映射,若字段为public并使用了@XmlTransient,get/set方法对不为public,需要在get/set方法上使用@XmlElement才会映射。

(2.2.4)XmlAccessType.NONE:

注:这一段的翻译是《任何字段或属性都不会绑定到XML,除非使用某些JAXB注释对它们进行特别注释。》

809e4caa78e9a9de96967ee47176de57.png

补充:

任何字段,get/set方法对都不会发生映射,除非使用某些注解,如@XmlElement,@XmlElementWrapper等。

(2.3)@XmlElement:

字段,方法,参数级别的注解。该注解可以将被注解的字段(非静态),或者被注解的get/set方法对应的字段映射为本地元素,也就是子元素。默认使用字段名或get/set方法去掉前缀剩下部分小写作为元素名(在字段名和get/set方法符合命名规范的情况下)。

属性:该注解的属性常用的属性有如下

(1)defaultValue:可以指定该元素默认的文本值

(2)namespace:可以指定该元素所属的命名空间

(3)name: 同@XmlRootElement注解的name属性一样

(4)required:可以指定该元素是否必须出现,默认为false

(5)nillable: 可以指定元素的文本值是否可以为空,默认为false

dd2a9c18bed31ff3702e338421af929c.png

15f2fd0bd8605dc2802349cf096c867f.png

(2.4)@XmlAttribute:

字段和方法级别的注解。该注解会将字段或get/set方法对应的字段映射成本类对应元素的属性,属性名默认使用字段名或get/set方法去掉前缀剩下部分首字母小写(在字段名和get/set方法符合命名规范的情况下)。修改上面例子:

属性:该注解有name,required,namespace三个属性。用法和@XmlElement注解相同

014cde630c516fc3820d9f9fcd1527c0.png

b3c090d099216b9e9bf446fe05d6eb17.png

(2.5)@XmlAccessorOrder:

包和类级别的注解。控制生成元素的顺序。

属性:该属性有XmlAccessOrder.ALPHABETICAL 和 XmlAccessOrder.UNDEFINED两种

(1)XmlAccessOrder.ALPHABETICAL,代表按照字母表的顺序对生成的元素排序,也就是我们常说的字典顺序

(2)XmlAccessOrder.UNDEFINED,代表按照类中字段的顺序生成元素的顺序,也是该注解的默认值

(2.6)@XmlElementWrapper

字段和方法级别的注解:围绕被映射的xml元素生成包装元素。主要用在集合对象映射后生成包装映射结果的xml元素,来看一下例子,创建两个类Children、Father。

e28add86e34561e651e41cb2c137617b.png

68f20f7148a94facf0c7093e83eb2b24.png

执行测试代码:

8d5758e2a4c9f03908193fd9ad9f3dd0.png

先来看一下没有@XmlElementWrapper注解的结果:

6b8a6d82438272d3f72d67b0f9e8aecb.png

修改一下代码再来看一下加了 @XmlElementWrapper注解的效果:

7d623986f66c07acd706eafb5cde1af3.png

打印结果:

4381efefb33ccc34df8f28ac0243f91d.png

(2.7)@XmlJavaTypeAdapter

包、类、字段,方法、参数级别的注解:解决java日期(Date),数字(Number)格式化问题。直接看例子,修改Person类,添加一个Date类型字段:

在这里要向使用@XmlJavaTypeAdapter我们就要指定一个指向将值类型转换为绑定类型的类,这个类需要继承XmlAdapter抽象类重写里面的unmarshal和marshal方法

b457e89322e8854477ebae215b3363d7.png

8e094141915eeeef492c69bcc573dda2.png

测试结果:

ecfae063b6c15eaf0f5125c2f8adc846.png

(2.8)@XmlTransient:

类,字段,方法级别的注解:可使JAXB在映射xml元素时忽略被注解的类,字段,get/set对应字段。需要注意的是该注解与所有其他JAXB注释相互排斥,也就是说与其他注释连用就会报错

5429e10960894b9e3e0db98a015c13c3.png

测试结果:

fea666234dfc1aa7de52dc0ed8095699.png

8b5be5349a8fee92c4e0543602b52234.png

原文:https://www.cnblogs.com/zouxiangzhongyan/p/10478114.html


http://chatgpt.dhexx.cn/article/rbHtiDQd.shtml

相关文章

JAXB应用实例

博文目录 一 JavaBean和XML相互转换初体验二 JAXB使用基础介绍三 应用实际四 问题补充 正文 过往的项目中数据存储都离不开数据库,不过最近做的一个项目的某些数据(比如人员信息、菜单、权限等等)却完全没有涉及任何数据库操作,直…

JAXB常用注解讲解(超详细)

简介: JAXB(Java Architecture for XML Binding) 是一个业界的标准,是一项可以根据XML Schema产生Java类的技术。该过程中,JAXB也提供了将XML实例文档反向生成Java对象树的方法,并能将Java对象树的内容重新写到XML实例…

OpenCV-Python (Canny边缘检测)

Canny边缘检测是一种非常流行的边缘检测算法,是John Canny在1986年提出的。它是一个多阶段的算法,即由多个步骤构成。 图像降噪计算图像梯度非极大值抑制阈值筛选 原型 OpenCV-Python中Canny函数的原型为: edge cv2.Canny(image, thresh…

python-opencv之边缘检测(cv2.Canny)

理论 Canny边缘检测是一种常用的边缘检测算法,并且他是一个多阶段的算法,分成如下几个阶段: 1. 消除噪声 由于边缘检测容易受到图像中噪声的影响,第一步是用5x5高斯滤波器去除图像中的噪声。 2. 寻找图像的强度梯度 然后对平滑后…

Canny边缘检测算法及实现

Canny边缘检测算法及实现 微信公众号:幼儿园的学霸 目录 文章目录 Canny边缘检测算法及实现目录前言原理步骤实现参考资料 前言 提取图片的边缘信息是底层数字图像处理的基本任务之一.边缘信息对进一步提取高层语义信息有很大的影响. 对图像提取边缘可以通过图像的…

Canny边缘检测算法

Canny是目前最优秀的边缘检测算法,其目标为找到一个最优的边缘,其最优边缘的定义为: 好的检测:算法能够尽可能的标出图像中的实际边缘好的定位:标识出的边缘要与实际图像中的边缘尽可能接近最小响应:图像中…

Canny算子边缘检测原理及实现

写在前面 Canny边缘检是在在1986年提出来的,到今天已经30多年过去了,但Canny算法仍然是图像边缘检测算法中最经典、先进的算法之一。 相比Sobel、Prewitt等算子,Canny算法更为优异。Sobel、Prewitt等算子有如下缺点: 没有充分利…

Opencv(C++)学习系列---Canny边缘检测算法

目录 【1】边缘检测算法流程 【2】Canny算子介绍 【3】完整代码 【1】边缘检测算法流程 使用高斯滤波器来平滑图像,达到滤除噪声的效果。(降噪)计算图像中每个像素点的梯度大小和方向(求梯度,与sobel算法求梯度一致)使用非极大值抑制&…

OpenCV-Canny边缘检测

作者:翟天保Steven 版权声明:著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处 函数原型 void Canny( InputArray image, OutputArray edges,double threshold1, double threshold2,int apertureSize 3, boo…

Python cv.Canny()方法参数与用法详解

函数原型与参数详解 OpenCV提供了cv.Canny()方法,该方法将输入的原始图像转换为边缘图像。该方法的原型为: cv.Canny(image, threshold1, threshold2[, edges[, apertureSize[, L2gradient]]]) -> edges cv.Canny(dx, dy, threshold1, threshold2[…

OpenCV—python 边缘检测(Canny)

边缘检测 一、边缘定义及类型二、边缘检测算子类别三、OpenCV-Python 中 Canny() 参数 一、边缘定义及类型 边缘类型:简单分为4中类型,阶跃型、屋脊型、斜坡型、脉冲型,其中阶跃型和斜坡型是类似的,只是变化的快慢不同。 二、边…

Canny算子Matlab实现

1、Canny边缘提取原理 Canny边缘检测方法利用了梯度方向信息,采用“非极大抑制”以及双阈值技术,获得了单像素连续边缘,是目前所认为的检测效果较好的一种边缘检测方法。 先利用高斯函数对图像进行低通滤波;然后对图像中的每个像…

canny算子的python实现以及pytorch实现

canny算子的python实现以及pytorch实现 canny的python实现canny的pytorch实现 canny的python实现 参考Canny边缘检测算法(python 实现) import numpy as np import cv2 as cv from matplotlib import pyplot as pltdef smooth(image, sigma 1.4, length 5) :&qu…

Python 使用cv2.canny 进行图像边缘检测

CV2提供了提取图像边缘的函数canny。其算法思想如下: 1. 使用高斯模糊,去除噪音点(cv2.GaussianBlur)2. 灰度转换(cv2.cvtColor)3. 使用sobel算子,计算出每个点的梯度大小和梯度方向4. 使用非极…

计算机视觉中Canny算子详解

文章目录 前言一、Canny的实现步骤二、具体实现1.高斯平滑滤波2.计算梯度大小和方向3.非极大抑制4.双阈值(Double Thresholding)和滞后边界跟踪 总结 前言 Canny边缘检测是一种非常流行的边缘检测算法,是John Canny在1986年提出的。它是一个多阶段的算法&#xff0c…

Canny 边缘检测算法

目录 一、边缘检测的步骤 二、最优边缘定义 三、Canny边缘检测算法步骤 1.对图像进行灰度化: 2.对图像进行高斯滤波: 3. 计算梯度幅值和方向 4.非极大值(Non-Maximum Suppression)抑制 5.用双阈值算法检测和连接边缘 代…

Canny算法

Canny Canny分为5个步骤 1)、使用高斯滤波器,以平滑图像,滤除噪声。 高斯滤波器是一种平滑空间滤波器,用于模糊处理和降低噪声。我们的高斯滤波器通过以下公式得到。 我们运用该公式计算出高斯卷积核,如k越大,检测…

OpenCv之Canny

目录 一、自适应阈值 二、边缘检测Canny 一、自适应阈值 引入前提:在前面的部分我们使用是全局闻值,整幅图像采用同一个数作为闻值。当时这种方法并不适应与所有情况,尤其是当同一幅图像上的不同部分的具有不同亮度时。这种情况下我们需要采用自适应闻…

图像处理——Canny算子

首先感谢以下两位的渊博知识: (1)爱鱼 https://www.cnblogs.com/mightycode/p/6394810.html (2)mitutao https://www.cnblogs.com/love6tao/p/5152020.html 图像边缘信息主要集中在高频段,通常…

Canny算子与霍夫变换检测圆与直线

目录 引言 一、canny算子 二、canny算子代码 三、霍夫变换检测直线 四、霍夫变换检测直线代码 五、霍夫变换检测直线效果 六、霍夫变换检测圆 七、霍夫变换检测圆代码 八、霍夫变换检测圆效果 引言 canny算子是计算机视觉最常用的一种算子,是目前一种非常流行…