JAXB中的@XmlRootElement注解详解

时间:2023-03-09 14:20:51
JAXB中的@XmlRootElement注解详解
  1. @Retention(value=RUNTIME)
  2. @Target(value=TYPE)
  3. public @interface XmlRootElement
  4. @Inherited
  5. @Retention(value=RUNTIME)
  6. @Target(value={PACKAGE,TYPE})
  7. public @interface XmlAccessorType

XmlRootElement: 将类或枚举类型映射到 XML 元素。JAXB中的注解,用来根据java类生成xml内容。

当使用 @XmlRootElement 注释对顶层类或枚举类型进行注释时,类型值被表示为 XML 文档中的 XML 元素。

JAXB Annotation

  • @XmlRootElement   // xml 文件的根元素
  • @XmlElement
  • @XmlAccessorType  // 控制默认情况下是否对字段或 Javabean 属性进行系列化。
  • @XmlTransient
  • @XmlJavaTypeAdaptor:参考Using JAXB 2.0's XmlJavaTypeAdapter

XmlAccessorType  
默认规则: 
默认情况下,如果包中不存在 @XmlAccessorType,那么假定使用以下包级别注释。 
@XmlAccessorType(XmlAccessType.PUBLIC_MEMBER) 
默认情况下,如果类中不存在 @XmlAccessorType,并且没有任何超类是使用 @XmlAccessorType 注释的,则假定在类中使用以下默认注释: 
@XmlAccessorType(XmlAccessType.PUBLIC_MEMBER) 
可能值: 
FIELD:    JAXB 绑定类中的每个非静态、非瞬态字段将会自动绑定到 XML,除非由 XmlTransient 注释。 
NONE:     所有字段或属性都不能绑定到 XML,除非使用一些 JAXB 注释专门对它们进行注释。 
PROPERTY: JAXB 绑定类中的每个获取方法/设置方法对将会自动绑定到 XML,除非由 XmlTransient 注释。 
PUBLIC_MEMBER:每个公共获取方法/设置方法对和每个公共字段将会自动绑定到 XML,除非由 XmlTransient 注释。

应用参考:http://blog.sina.com.cn/s/blog_4051f5dc0100ju0a.html

  1. package jaxb;
  2. import javax.xml.bind.annotation.XmlAccessorType;
  3. import javax.xml.bind.annotation.XmlRootElement;
  4. import javax.xml.bind.annotation.XmlAccessType;
  5. @XmlRootElement // 必须要标明这个元素
  6. @XmlAccessorType(XmlAccessType.FIELD)
  7. public class Boy {
  8. String name = "CY";
  9. }
  10. package jaxb;
  11. import java.io.StringReader;
  12. import javax.xml.bind.JAXBContext;
  13. import javax.xml.bind.JAXBException;
  14. import javax.xml.bind.Marshaller;
  15. import javax.xml.bind.Unmarshaller;
  16. public class JAXBTest {
  17. public static void main(String[] args) throws JAXBException {
  18. JAXBContext context = JAXBContext.newInstance(Boy.class);
  19. Marshaller marshaller = context.createMarshaller();
  20. Unmarshaller unmarshaller = context.createUnmarshaller();
  21. Boy boy = new Boy();
  22. marshaller.marshal(boy, System.out);
  23. System.out.println();
  24. String xml = "<boy><name>David</name></boy>";
  25. Boy boy2 = (Boy) unmarshaller.unmarshal(new StringReader(xml));
  26. System.out.println(boy2.name);
  27. }
  28. }

执行结果: 
<?xml version="1.0" encoding="UTF-8" standalone="yes"?><boy><name>CY</name></boy> 
David 
(先 是marshall成 xml文件,再是把 xml 文件 unmarshal 成 java object。)

改动一: 
修改@XmlAccessorType(XmlAccessType.FIELD)  --> @XmlAccessorType(XmlAccessType.PROPERTY) 
意思是只有 属性 才能被转换成 xml 中的标签。

所以再运行的结果是: 
<?xml version="1.0" encoding="UTF-8" standalone="yes"?><boy/> 
CY 
就是说 java object 转换成 xml 的时候,name 不是属性(因为没有 get set方法),所以name不转换成标签。

改动二: 
在 改动一 的基础上,给name属性添加 get set 方法。 再运行,结果为: 
<?xml version="1.0" encoding="UTF-8" standalone="yes"?><boy><name>CY</name></boy> 
David 
由 此 可见 @XmlAccessorType 这个annotation 的作用。

改动三: 
在改动二 的基础上,给Boy 再添加一个field, int age=10, 即:

  1. package jaxb;
  2. import javax.xml.bind.annotation.XmlAccessorType;
  3. import javax.xml.bind.annotation.XmlRootElement;
  4. import javax.xml.bind.annotation.XmlAccessType;
  5. @XmlRootElement
  6. @XmlAccessorType(XmlAccessType.PROPERTY)
  7. public class Boy {
  8. String name = "CY";
  9. int age = 10;
  10. public String getName() {
  11. return name;
  12. }
  13. public void setName(String name) {
  14. this.name = name;
  15. }
  16. }

显然,这个age 是不会被 转化 到xml 文件中的。解决办法是:

  1. import javax.xml.bind.annotation.XmlAccessorType;
  2. import javax.xml.bind.annotation.XmlElement;
  3. import javax.xml.bind.annotation.XmlRootElement;
  4. import javax.xml.bind.annotation.XmlAccessType;
  5. @XmlRootElement // bixude
  6. @XmlAccessorType(XmlAccessType.PROPERTY)
  7. public class Boy {
  8. String name = "CY";
  9. @XmlElement
  10. int age = 10;
  11. public String getName() {
  12. return name;
  13. }
  14. public void setName(String name) {
  15. this.name = name;
  16. }
  17. }

加上 @XmlElement annotation. 运行结果为: 
<?xml version="1.0" encoding="UTF-8" standalone="yes"?><boy><age>10</age> <name>CY</name></boy> 
David

对于根元素,可以设置属性: 
@XmlRootElement(name="b" nameSpace="http://test") 
这样,在生成的xml文件中,<boy> 标签 就会变为 <b> 标签。并且加上一个命名空间。

下面解释 @XmlJavaTypeAdaptor 的作用:

  1. @XmlRootElement
  2. @XmlAccessorType(XmlAccessType.PROPERTY)
  3. public class Boy {
  4. private String name = "CY";
  5. private Address address; // 是一个接口
  6. public String getName() {
  7. return name;
  8. }
  9. public void setName(String name) {
  10. this.name = name;
  11. }
  12. }

在 java object 转换成 xml 的时候,接口Address 无法被转换。 
所以 这里要加上 @XmlJavaTypeAdapter(AddressAdapter.class) 
所以 要多写一个AddressAdaptor 类。 
这个类会返回Address接口的一个具体实现类的对象。

这 就是 @XmlJavaTypeAdapter 的作用。