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

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

forJL

简介:

JAXB(Java Architecture for XML Binding) 是一个业界的标准,是一项可以根据XML Schema产生Java类的技术。该过程中,JAXB也提供了将XML实例文档反向生成Java对象树的方法,并能将Java对象树的内容重新写到XML实例文档。从另一方面来讲,JAXB提供了快速而简便的方法将XML模式绑定到Java表示,从而使得Java开发者在Java应用程序中能方便地结合XML数据和处理函数。

 

一个简单的例子

Person.class :

 
  1. @XmlRootElement

  2. public class Person {

  3. private int id;

  4. private String name;

  5. private String gender;

  6. private String addr;

  7.     private String area;

  8.  
  9. public Person() {

  10. }

  11.  
  12. public Person(String name, String gender, String addr, String area) {

  13. this.name = name;

  14. this.gender = gender;

  15. this.addr = addr;

  16. this.area = area;

  17. }

  18.  
  19. public int getId() {

  20. return id;

  21. }

  22.  
  23. @XmlElement

  24. public void setId(int id) {

  25. this.id = id;

  26. }

  27.  
  28. public String getName() {

  29. return name;

  30. }

  31.  
  32. @XmlElement

  33. public void setName(String name) {

  34. this.name = name;

  35. }

  36.  
  37. public String getGender() {

  38. return gender;

  39. }

  40.  
  41. @XmlElement

  42. public void setGender(String gender) {

  43. this.gender = gender;

  44. }

  45.  
  46. public String getAddr() {

  47. return addr;

  48. }

  49.  
  50. @XmlElement

  51. public void setAddr(String addr) {

  52. this.addr = addr;

  53. }

  54.  
  55. public String getArea() {

  56. return area;

  57. }

  58.  
  59. @XmlElement

  60. public void setArea(String area) {

  61. this.area = area;

  62. }

  63.  
  64. @Override

  65. public String toString() {

  66. return "Person{" +

  67. "id=" + id +

  68. ", name='" + name + '\'' +

  69. ", gender='" + gender + '\'' +

  70. ", addr='" + addr + '\'' +

  71. ", area='" + area + '\'' +

  72. '}';

  73. }

  74. }

Test:

 
  1. public class JAXBTest {

  2. @Test

  3. public void generateXML() {

  4. Person person = new Person("abc", "男", "北京", "朝阳区");

  5.  
  6. File file = new File("E:\\person.xml");

  7. JAXBContext jc = null;

  8. try {

  9. //根据Person类生成上下文对象

  10. jc = JAXBContext.newInstance(Person.class);

  11. //从上下文中获取Marshaller对象,用作将bean编组(转换)为xml

  12. Marshaller ma = jc.createMarshaller();

  13. //以下是为生成xml做的一些配置

  14. //格式化输出,即按标签自动换行,否则就是一行输出

  15. ma.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);

  16. //设置编码(默认编码就是utf-8)

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

  18. //是否省略xml头信息,默认不省略(false)

  19. ma.setProperty(Marshaller.JAXB_FRAGMENT, false);

  20.  
  21. //编组

  22. ma.marshal(person, file);

  23. } catch (JAXBException e) {

  24. e.printStackTrace();

  25. }

  26. }

  27.  
  28. @Test

  29. public void generateBean() {

  30. File file = new File("E:\\person.xml");

  31. JAXBContext jc = null;

  32. try {

  33. jc = JAXBContext.newInstance(Person.class);

  34. Unmarshaller uma = jc.createUnmarshaller();

  35. Person person = (Person) uma.unmarshal(file);

  36. System.out.println(person);

  37. } catch (JAXBException e) {

  38. e.printStackTrace();

  39. }

  40. }

  41. }

 

测试结果:

generateXML():

 

 

此xml相当于该xsd文件:

 
  1. <xs:element name="person">

  2. <xs:complexType>

  3. <xs:sequence>

  4. <xs:element type="xs:string" name="addr" minOccurs="0"/>

  5. <xs:element type="xs:string" name="area" minOccurs="0"/>

  6. <xs:element type="xs:string" name="gender" minOccurs="0"/>

  7. <xs:element type="xs:int" name="id" minOccurs="0"/>

  8. <xs:element type="xs:string" name="name" minOccurs="0"/>

  9. </xs:sequence>

  10. </xs:complexType>

  11. </xs:element>

generateBean():

是不是很方便?这只是一个最简单的小例子,下文会在这个例子的基础上介绍和讲解其他的一些常用JAXB注解。因为会使用到部分xsd的知识,不了解的读者可以看我另一篇博客:《xsd学习:超详细解析》,否则下文会有些理解困难。

注:从jdk1.7开始,JAXB就对解组和编组的方法进行了更简单的封装,所以实际项目中除非自己要进行个性化设置,否则大可不用自己再创建JAXBContext实例,直接通过JAXB静态调用相应的工具方法就行了,于是上面的测试方法可以写的更简练些:

 
  1. public class JAXBTest {

  2. @Test

  3. public void generateXML() {

  4. Person person = new Person("abc", "男", "北京", "朝阳区");

  5. File file = new File("E:\\person.xml");

  6. JAXB.marshal(person, file);

  7. }

  8.  
  9. @Test

  10. public void generateBean() {

  11. File file = new File("E:\\person.xml");

  12. Person person = JAXB.unmarshal(file, Person.class);

  13.  
  14. System.out.println(person);

  15. }

  16. }

 

直接使用默认的配置,已经足够应付大多数情况,读者可以试一下。

 

常用注解:

@XmlRootElement:

作用和用法:

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

属性:

该注解含有name和namespace两个属性。namespace属性用于指定生成的元素所属的命名空间。name属性用于指定生成元素的名字,若不指定则默认使用类名小写作为元素名。修改上面的例子,在该注解上使用name属性:

 
  1. @XmlRootElement(name = "human")

  2. public class Person {

  3. private int id;

  4. private String name;

  5.     ...

  6.     ...

  7. }    //省略下面代码

生成的xml:

 

@XmlElement

作用和用法:

字段,方法,参数级别的注解。该注解可以将被注解的字段(非静态),或者被注解的get/set方法对应的字段映射为本地元素,也就是子元素。默认使用字段名或get/set方法去掉前缀剩下部分小写作为元素名(在字段名和get/set方法符合命名规范的情况下)。上面例子中,id、addr、name、gender、area都被映射成了<person>元素的子元素。下文会配合@XmlAccessorType注解详细讲解该注解的用法。常与@XmlValue,@XmlJavaTypeAdapter,@XmlElementWrapper连用。

属性:

该注解的属性常用的属性有有:name、nillable、required、namespace、defaultValue

* name属性可以指定生成元素的名字,同@XmlRootElement注解的name属性一样,不再举例。

* nillable属性可以指定元素的文本值是否可以为空,默认为false。修改上面例子:

 
  1. @XmlElement(nillable = true)

  2. public void setName(String name) {

  3.     this.name = name;

  4. }

则生成的xsd(为了节省篇幅,只截取必要的片段)为:

<xs:element name="name" type="xs:string" nillable="true" minOccurs="0"/>

* required属性可以指定该元素是否必须出现,默认为false,所以在xsd中会有对应的属性minOccurs="0"。修改该属性为true

 
  1. @XmlElement(nillable = true, required = true)

  2. public void setName(String name) {

  3. this.name = name;

  4. }

生成的xsd文件为:

<xs:element name="name" type="xs:string" nillable="true" minOccurs="1"/>

* namespace属性可以指定该元素所属的命名空间

* defaultValue属性可以指定该元素默认的文本值

 

@XmlAttribute

作用和用法:

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

 
  1. @XmlAttribute

  2. public void setGender(String gender) {

  3. this.gender = gender;

  4. }

生成的xml:

对应的xsd:

 
  1. <xs:element name="human">

  2. <xs:complexType>

  3. <xs:sequence>

  4. <xs:element type="xs:string" name="addr"/>

  5. <xs:element type="xs:string" name="area"/>

  6. <xs:element type="xs:byte" name="id"/>

  7. <xs:element type="xs:string" name="name"/>

  8. </xs:sequence>

  9. <xs:attribute type="xs:string" name="gender"/>

  10. </xs:complexType>

  11. </xs:element>

属性:

该注解有name,required,namespace三个属性。用法和@XmlElement注解相同,不再举例,可以自己尝试下。

 

@XmlTransient

作用和用法:

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

 
  1. @XmlTransient

  2. public void setId(int id) {

  3. this.id = id;

  4. }

生成的xml:

属性:

 

该注解没有属性。

 

@XmlAccessorType

作用和用法:

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

属性:

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

XmlAccessType.PROPERTY:

官方解释:

Every getter/setter pair in a JAXB-bound class will be automatically bound to XML, unless annotated by {@link XmlTransient}.
jaxb绑定类中的每个getter/setter对都将自动绑定到XML,除非用@XmlTransient注释。
Fields are bound to XML only when they are explicitly annotated by some of the JAXB annotations.
只有在某些JAXB注释显式地注释字段时,字段才被绑定到XML。

补充:

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元素。

XmlAccessType.FIELD:

官方解释:

Every non static, non transient field in a JAXB-bound class will be automatically bound to XML, unless annotated by {@link XmlTransient}.
jaxb绑定类中的每个非静态、非瞬态字段都将自动绑定到XML,除非使用@XmlTransient进行注释。
Getter/setter pairs are bound to XML only when they are explicitly annotated by some of the JAXB annotations.
只有当某些JAXB注释显式地对getter/setter对进行注释时,它们才会绑定到XML。

补充:

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

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

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

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

官方解释:

Every public getter/setter pair and every public field will be automatically bound to XML, unless annotated by {@link XmlTransient}.
每个公共getter/setter对和每个公共字段都将自动绑定到XML,除非使用@XmlTransient注释。
Fields or getter/setter pairs that are private, protected, or defaulted to package-only access are bound to XML only when they areexplicitly annotated by the appropriate JAXB annotations.
只有在适当的JAXB注释显式地注释了字段或getter/setter对之后,才会将它们绑定到XML。

补充:

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

2.若使用@XmlElement注解,需要注意只能在字段或get/set方法添加,两者任选其一,否则会报属性重复的错误。

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

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

XmlAccessType.NONE:

官方解释:

None of the fields or properties is bound to XML unless they are specifically  annotated with some of the JAXB annotations.
任何字段或属性都不会绑定到XML,除非使用某些JAXB注释对它们进行特别注释。

补充:

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

 

 

@XmlAccessorOrder

作用和用法:

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

属性:

只有一个value属性,可取的值是一个名为XmlAccessOrder的枚举类型的两个值,XmlAccessOrder.ALPHABETICAL 和 XmlAccessOrder.UNDEFINED。默认为XmlAccessOrder.UNDEFINED,代表按照类中字段的顺序生成元素的顺序。

另一个值则代表按照字母表的顺序对生成的元素排序。但奇怪的是,只有jaxb按照field生成元素时,默认值才会生效,否则总是按照字母表的顺序排序。

 

@XmlElementWrapper

作用和用法:

字段和方法级别的注解。围绕被映射的xml元素生成包装元素。主要用在集合对象映射后生成包装映射结果的xml元素。 

修改上面的例子,添加一个Key类,在Person类中添加一个Key类的Set集合,修改如下:

Key类:

 
  1. @XmlRootElement

  2. public class Key {

  3. private String roomNum;

  4.  
  5. public Key() {

  6. }

  7.  
  8. public Key(String roomNum) {

  9. this.roomNum = roomNum;

  10. }

  11.  
  12. public String getRoomNum() {

  13. return roomNum;

  14. }

  15.  
  16. @XmlElement

  17. public void setRoomNum(String roomNum) {

  18. this.roomNum = roomNum;

  19. }

  20.  
  21. @Override

  22. public String toString() {

  23. return "Key{" +

  24. "roomNum='" + roomNum + '\'' +

  25. '}';

  26. }

  27. }

Person类:

 
  1. @XmlRootElement(name = "human")

  2. @XmlAccessorType(XmlAccessType.PUBLIC_MEMBER)

  3. @XmlAccessorOrder(XmlAccessOrder.ALPHABETICAL)

  4. public class Person {

  5. ...

  6. private Set<Key> key = new HashSet<>();

  7.  
  8. public Person() {

  9. }

  10.  
  11. public Person(String name, String gender, String addr, String area) {

  12. this.name = name;

  13. this.gender = gender;

  14. this.addr = addr;

  15. this.area = area;

  16. key.add(new Key("001"));    //向集合中添加两个Key对象

  17. key.add(new Key("002"));

  18. }

  19.     ...

  20.  
  21. public Set<Key> getKey() {

  22. return key;

  23. }

  24.  
  25. @XmlElement

  26. public void setKey(Set<Key> key) {

  27. this.key = key;

  28. }

  29. }

生成的xml:

但同一元素应该需要被“包装”一下才显得有层次感,所以可以使用@XmlElementWrapper来实现:

 
  1. @XmlElementWrapper(name = "keys")

  2. @XmlElement

  3. public void setKey(Set<Key> key) {

  4. this.key = key;

  5. }

生成的xml:

这样是不是好多了?

属性:

该注解有name、nillable、namespace、required四个属性,用法同上,不再赘述。

 

 

@XmlJavaTypeAdapter

作用和用法:

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

Person类:

 
 
  1. @XmlRootElement(name = "human")

  2. /*@XmlType(propOrder = {

  3. "name",

  4. "gender",

  5. "addr",

  6. "area"

  7. })*/

  8. @XmlAccessorType(XmlAccessType.PUBLIC_MEMBER)

  9. @XmlAccessorOrder(XmlAccessOrder.ALPHABETICAL)

  10. public class Person {

  11. ...

  12. private Date date = new Date();

  13.  
  14.     ...

  15.  
  16. public Date getDate() {

  17. return date;

  18. }

  19.  
  20. @XmlElement

  21. public void setDate(Date date) {

  22. this.date = date;

  23. }

  24. }

生成的xml:

 

 

 

这样的date格式显然令人不满意,我们需要例如“2018-05-20”这样的格式。这就需要@XmlJavaTypeAdapter注解,自定义一个适配器来解决这个问题。该注解的用法就是自定义适配器并继承XmlAdapter类,实现里面的marshal和unmarshal方法,并在该注解上引用。修改例子:

 

自定义的DateAdapter:

 
  1. public class DateAdapter extends XmlAdapter<String, Date> {

  2. private static final SimpleDateFormat SDF = new SimpleDateFormat("yyyy-MM-dd");

  3.  
  4. @Override

  5. public Date unmarshal(String date) throws Exception {

  6. return SDF.parse(date);

  7. }

  8.  
  9. @Override

  10. public String marshal(Date date) throws Exception {

  11. return SDF.format(date);

  12. }

  13. }

Person类:

 
  1. @XmlJavaTypeAdapter(DateAdapter.class)

  2. @XmlElement

  3. public void setDate(Date date) {

  4. this.date = date;

  5. }

 

生成的xml:

 

 

完美解决。

 

属性:

常用的就是value属性,其他属性请自行研究。

 

 

@XmlValue:

 

 

作用和用法:

 

字段和方法级别的注解。该注解的作用,简单理解就是定义xml元素文本值的类型,例如在一个类的String类型字段上使用该注解,则生成的元素文本值类型就是xsd:string,也就是定义一个xsd中的simpleType.若类中还有一个字段并使用了@XmlAttribute注解,则是定义一个xsd中的complexType。

 

属性:

 

 

 

@XmlType:

 

 

作用和用法:

 

类级别的注解。该注解有些复杂,主要使用的是它的propOrder属性,简单来说是用来定义xsd中的simpleType或complexType,从生成的xml中来看,它的作用就是指定生成元素的顺序,具体看下图:

 

 

简单解释下每行什么意思:

 

* 若指定该注解的propOrder为{},会生成ComplexType并且使用xs:all指示器,表示所有被映射的元素都必须出现在xml中

 

* 若propOrder的值为{"name", "addr", "area"}(大括号中都是Person类的字段名称),会生成ComplexType并使用xs:sequence指示器,表示生成的xml元素必须按照propOrder指定的顺序出现,也就间接实现了排序。

 

* 若不指定propOrder属性(这与指定propOrder但值为{}不同),没有字段,会生成ComplexType并包含一个空的xs:sequence指示器。

 

* 若不指定propOrder属性,但含有被@XmlValue注解的字段和被@XmlAttribute注解的字段,会生成一个含有simpleContent的ComplexType。

 

* 若不指定propOrder属性,但含有被@XmlValue注解的字段而没有被@XmlAttribute注解的字段,会生成一个含有simpleType的ComplexType。

再次强调,如果以上解释看不懂,先学xsd。传送门:《XSD学习》

评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值