JAXB学习(二): 对JAXB支持的主要注解的说明

我们在上一篇中对JAXB有了一个大致的认识,现在我们来了解JAXB的一些主要注解。

 

顶层元素:XmlRootElement

表示整个XML文档的类应该使用XmlRootElement修饰,其实就像之前那个简单例子那样,XmlRootElement也是最简单JAXB应用唯一需要用到的注解。

 

控制元素的选择:XmlAccessorType XmlTransient

如果JAXB绑定一个java类为XML,那么默认的会绑定所有public成员,包括 public的getter和setter对(必须同时有getter和setter)或者是public的属性。任何protected ,default和private的成员只有在被一个恰当的注解(例如 XmlElement 或者XmlAttribute)修饰时才会被绑定。 我们有几种方式来影响这种默认的行为。

1. 在 包 或者 顶层元素(也就是XmlRootElement修饰的类)上 使用 XmlAccessorType, 它的值有  FIELDPROPERTYPUBLIC_MEMBER or NONE

  FIELD : 任何非static 非 transient 的属性将会被绑定

  PROPERTY :任何getter和setter对

  PUBLIC_MEMBER : 这个就是上面描述的默认情况

  NONE : NONE会压制任何绑定,除非明确的使用XmlElement或XmlAttribute修饰。

没有这个注解的类 可以从父类或者包级别的配置来继承。

2. 使用XmlTransient, 它会压制它的目标绑定。 考虑下面这种情况,有一个public 的属性foo,还有一对getFoo和setFoo,如果我们使用默认的配置将会出现 命名冲突,这时就可以使用XmlTransient来压制其中一个。

下面的例子我们在包级别使用XmlAccessorType,将绑定设置为 FIELD, 然后使用XmlTransient 压制其中的一个public属性。

首先在包下面建一个package-info.java 文件用来进行包注释。

@javax.xml.bind.annotation.XmlAccessorType(javax.xml.bind.annotation.XmlAccessType.FIELD)
package com.massclouds.test;

 

顶层元素类:

package com.massclouds.test;

import javax.xml.bind.annotation.XmlRootElement;
import javax.xml.bind.annotation.XmlTransient;

@XmlRootElement
public class Person {
    public String name;
    @XmlTransient
    public int age;

    private String gender;

    public String getGender() {
        return gender;
    }
    public void setGender(String gender) {
        this.gender = gender;
    }
}

 

在上面的配置中,Person的name和gender属性可以被绑定到xml中。 注意gender之所以会被绑定并不是因为getter和setter对,而是应为FIELD级别会将private的属性也绑定。

3.使用XmlElement和XmlAttribute 

 我们可以使用这两个元素来打破XmlAccessorType的规则,主动要求绑定。 例如默认情况下,private的属性是不会被绑定的,下面我们分别使用这两个注解来注释两个私有的属性。

import javax.xml.bind.annotation.XmlAttribute;
import javax.xml.bind.annotation.XmlElement;
import javax.xml.bind.annotation.XmlRootElement;

@XmlRootElement
public class Person {
    @XmlAttribute
    private String id;
    @XmlElement
    private String name;
    
    //这个无参构造器是必须的
    public Person(){}
    
    public Person(String id, String name){
        this.name = name;
        this.id = id;
    }
}

 

使用JAXB序列化一个Person对象的结果为:

<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<person id="person_001">
    <name>zhangsan</name>
</person>

另外还有一个XmlElements注解。 考虑下面情况:

//这里不能使用接口, JAXB 无法处理接口
public abstract class Animal {}

public class Cat extends Animal{
    public String color = "red";
}

public class Dog extends Animal{
    public String size = "big";
}

public class Pig extends Animal{
    public String weight = "200kg";
}

//Person类是JAXB根元素,包含一个Animal的集合
@XmlRootElement
public class Person {
    public List<Animal> animals = Arrays.asList(new Dog(), new Cat(), new Pig());
}

 

我们对一个Person对象序列化后的结构是:

<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<person>
    <animals/>
    <animals/>
    <animals/>
</person>

 这很显眼不是我们所希望的,这是因为JAXB会将animals集合中的每一个元素仅仅当做Animal来处理,而不会考虑各种子类的具体情况。我们现在使用XmlElements来修饰animals,让JAXB区别对待各种Animal的子类。

@XmlRootElement
public class Person {
    @XmlElements({
        @XmlElement(name="dog", type=Dog.class),
        @XmlElement(name="cat", type=Cat.class),
        @XmlElement(name="pig", type=Pig.class)
    })
    public List<Animal> animals = Arrays.asList(new Dog(), new Cat(), new Pig());
}

 

 结果为:

<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<person>
    <dog>
        <size>big</size>
    </dog>
    <cat>
        <color>red</color>
    </cat>
    <pig>
        <weight>200kg</weight>
    </pig>
</person>

 也许你发现了上面序列化生成的xml并不是最理想的方式,因为对于person对象来讲,它拥有一个animals的集合,可是这个结果却体现不出来这一点。对于集合类型,我们可以使用@XmlElementWrapper来给集合属性增加一个wrapper。 修改Person为:

@XmlRootElement
public class Person {
    @XmlElementWrapper(name="animals")
    @XmlElements({
        @XmlElement(name="dog", type=Dog.class),
        @XmlElement(name="cat", type=Cat.class),
        @XmlElement(name="pig", type=Pig.class)
    })
    public List<Animal> animals = Arrays.asList(new Dog(), new Cat(), new Pig());
}

 

 现在的结果就更加理想了:

<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<person>
    <animals>
        <dog>
            <size>big</size>
        </dog>
        <cat>
            <color>red</color>
        </cat>
        <pig>
            <weight>200kg</weight>
        </pig>
    </animals>
</person>

 

随机属性和随机元素: @XmlAnyAttribute @XmlAnyElement 

我们之前的例子中序列化后产生的xml中无论是attribute还是element 在 java类中都会有具体的 Filed或者 getter/setter 与之对应,但是如果在序列化时我们无法确定java对象到底会有哪些属性,或者 在反序列化的过程中xml文档中的内容是不确定的,我们该怎么办呢?  我们可以使用XmlAnyAttribute 和XmlAnyElement来支持任意随机的attribute和element。

import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import javax.xml.bind.annotation.XmlAnyAttribute;
import javax.xml.bind.annotation.XmlAnyElement;
import javax.xml.bind.annotation.XmlRootElement;
import javax.xml.namespace.QName;
import org.w3c.dom.Element;

@XmlRootElement
public class Person {
    public String name;
    
    @XmlAnyAttribute
    public Map<QName, String> anyAttribute = new HashMap<>();
    
    @XmlAnyElement
    public List<Element> anyElement = new ArrayList<>();
}

现在根元素 Person 就可以支持任意的属性和元素了,值得注意的是XmlAnyElement注释了一个org.w3c.dom.Element的集合,其实就是在处理最原始的dom元素了。 下面是序列化的过程:

import java.io.FileOutputStream;
import javax.xml.bind.JAXBContext;
import javax.xml.bind.JAXBException;
import javax.xml.bind.Marshaller;
import javax.xml.namespace.QName;
import javax.xml.parsers.DocumentBuilderFactory;
import javax.xml.parsers.ParserConfigurationException;
import org.w3c.dom.Document;
import org.w3c.dom.Element;

public class Test {
    public static void main(String[] args) throws JAXBException, ParserConfigurationException {
        JAXBContext context = JAXBContext.newInstance(Person.class);
        Marshaller marshaller = context.createMarshaller();
        marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);
        
        Person person = new Person();
        person.name = "zhangsan";
        person.anyAttribute.put(new QName("", "id"), "11");

        Document document = DocumentBuilderFactory.newInstance().newDocumentBuilder().newDocument();
        Element e1 = document.createElement("company");
        e1.setTextContent("www.massclouds.com");
        
        person.anyElement.add(e1);

        try(FileOutputStream out = new FileOutputStream("C:/temp/any.xml")){
            marshaller.marshal(person, out);
        }catch(Exception e){
            e.printStackTrace();
        }
    }
}

 

我们在相应的文件中产生了下面的xml内容:

<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<person id="11">
    <name>zhangsan</name>
    <company>www.massclouds.com</company>
</person>

 

我们对其进行修改,为person增加一个age属性和一个address子元素,变为下面这样:

<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<person id="11" age="23">
    <name>zhangsan</name>
    <company>www.massclouds.com</company>
    <address>Shandong Jinan</address>
</person>

 

然后我们对这个xml文档进行反序列化,得到的person对象中就可以包含所有这些随机属性和随机元素了。

public class UnTest {
    public static void main(String[] args) throws JAXBException {
        JAXBContext context = JAXBContext.newInstance(Person.class);
        Unmarshaller u = context.createUnmarshaller();
        
        try(FileInputStream in = new FileInputStream("C:/temp/any.xml")){
            Person person = (Person)u.unmarshal(in);
            //读取person的随机属性
            person.anyAttribute.forEach((key, value) -> System.out.println(key + "-->" + value));
            //读取person的随机元素
            person.anyElement.forEach(element -> System.out.println(element.getTagName() + "-->" + element.getTextContent()));
        }catch(Exception e){
            e.printStackTrace();
        }
    }
}

 

 

使用适配器来改变JAXB序列化规则: XmlJavaTypeAdapter

考虑这么一种场景,在java类中我们有一个StringBuffer类型的属性,很显然我们就是希望把它序列化为一个 简单字符串类型,但是JAXB默认是不支持StringBuffer的这种转换的。我们可以使用XmlJavaTypeAdapter来指定一个序列化的适配器,按照我们自己的逻辑来定制序列化规则。

@XmlRootElement
public class Person {
    @XmlElement
    @XmlJavaTypeAdapter(String2StrBuf.class)
    public StringBuffer poem = new StringBuffer();
    
    {
        this.poem = new StringBuffer();
        //from Dido's Everything to Lose
        this.poem.append("I love to be alive ").append("but I was not afraid to die");
    }
}

 

我们在根元素中使用XmlJavaTypeAdapter修饰了一个StringBuffer类型,其中value的值 String2StrBuf就是我们要自己定义的适配器,它必须继承自XmlAdapter。

import javax.xml.bind.annotation.adapters.XmlAdapter;

public class String2StrBuf extends XmlAdapter<String, StringBuffer> {
    @Override
    public String marshal(StringBuffer strbuf) {
        return strbuf.toString();
    }

    @Override
    public StringBuffer unmarshal(String string) {
        return new StringBuffer(string);
    }
}

 

从上面的方法签名中我们就可以看出在序列化的过程中调用marshal方法,反之则调用unmarshal。

我们还可以在包级别定义适配器,这样就不需要再包中重复的定义了(以下代码定义在package-info.java中)。

@javax.xml.bind.annotation.adapters.XmlJavaTypeAdapters({
    @javax.xml.bind.annotation.adapters.XmlJavaTypeAdapter(type=java.lang.StringBuffer.class, value=com.massclouds.test.String2StrBuf.class)
})
package com.massclouds.test;

 

 有了上面的包级别的定义,在Person中poem属性上的XmlJavaTypeAdapter就可以不要了。

下面我们在来演示一个使用JAXB自带的Adapter来序列化二进制数据的例子,我们将一副图片的二进制数据序列化到一个xml文件中,然后再从xml文件恢复这张图片(虽然现实中这么做有点无聊,^_^)

@XmlRootElement
public class Image {
    @XmlJavaTypeAdapter(HexBinaryAdapter.class)
    public byte[] data;
}

 

Image是顶层元素,它的data属性就是我们要序列化到xml文件中的二进制数据,我们看到它使用了HexBinaryAdapter适配器, 这个适配器的作用很明显: 将二进制数据绑定为16进制表示。

    public static void main(String[] args) {
        try(FileInputStream in = new FileInputStream("C:/6.png");
            FileOutputStream xmlOut = new FileOutputStream("C:/temp/data.xml");){
            ByteArrayOutputStream out = new ByteArrayOutputStream();
            byte[] buffer = new byte[1024];
            int len = 0;
            while((len = in.read(buffer, 0, buffer.length)) != -1){
                out.write(buffer, 0, len);
            }
            Image image = new Image();
            image.data = out.toByteArray();
            
            JAXBContext context = JAXBContext.newInstance(Image.class);
            Marshaller marshaller = context.createMarshaller();
            marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);
            marshaller.marshal(image, xmlOut);
        }catch (Exception e) {
            e.printStackTrace();
        }
    }

 

上面的代码除去处理流的代码外,剩下的就是JAXB最基本的序列化代码了,也就是说我们并没有做过多其他的处理。生成的xml文件为:

<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<image>
       <!--data的内容就是图片的16进制表示了 -->
     <data>191E7721C8A4C714514C10E3C30A777CD1454B18D3C
        .... 
        B52F7A28A64A15BA1FA5314EE539ED45142067FFD9</data>
</image>

 

我们再将xml反序列化,重新得到这幅图片,同样除去流的处理外,只是最基本的JAXB操作而已:

    public static void main(String[] args) {
        try(FileOutputStream out = new FileOutputStream("C:/temp/image.png");
                FileInputStream in = new FileInputStream("C:/temp/data.xml")){
            JAXBContext context = JAXBContext.newInstance(Image.class);
            Unmarshaller u = context.createUnmarshaller();
            Image image = (Image)u.unmarshal(in);
            out.write(image.data);
            
        }catch(Exception e){
            e.printStackTrace();
        }
    }

 

上面我们使用的是图片的byte数据,我们同样也可以结合java 序列化(Serializable)来使用,也就是将一个对象的java序列化数据写入到xml文件中。

对象引用:XmlIDXmlIDREF

考虑这样一种场景: 根元素是 教室, 在一个教室中有许多学生和一个老师(注意始终就只有这一个老师), 这些学生对象都包含一个老师属性,而且都指向教室中的老师, 这个老师同样也有许多学生,也就是教室中的这些学生。

如果我们按照正常的思路去一步一步实现上面这个描述,会出现两个问题:

1. 在序列化的时候将会产生很多重复数据(唯一一个老师却产生了许多老师数据),而在反序列化的时候我们将无法得到我们期待的结果(例如会产生不止一个老师)。

2. 还有可能出现死循环,导致无法序列化。

为了解决上面的问题,我们可以使用XmlID和XmlIDREF来引用对象。其中XmlID必须修饰一个String类型的属性。 下面是具体实现:

public class Teacher {
    @XmlID
    public String id;
    
    @XmlIDREF
    @XmlElementWrapper(name="students")
    @XmlElement(name="student_ref")
    public List<Student> students;
    
    public Teacher(){}
    public Teacher(String id){
        this.id = id;
    }
}

 

public class Student {
    @XmlID
    public String id;
    @XmlIDREF
    @XmlElement(name="teacher_ref")
    public Teacher teacher;
    
    public Student(){}
    
    public Student(String id, Teacher teacher){
        this.id = id;
        this.teacher = teacher;
    }
}

 

在上面的Teacher和Student中,分别使用XmlID定义了他们各自的ID属性,然后在Student中使用XmlIDREF引用一个Teacher, 而在Teacher中使用XmlIDREF引用一个Student的集合。

@XmlRootElement
public class Classroom {
    @XmlElementWrapper(name="students")
    @XmlElement(name="student")
    public List<Student> students;
    
    public Teacher teacher;
}

 

根元素没有使用 XmlIDREF的原因是我们必须让老师和每一个学生至少完整的出现一次。

public class Test {
    public static void main(String[] args) throws JAXBException {
        JAXBContext context = JAXBContext.newInstance(Classroom.class);
        Marshaller marshaller = context.createMarshaller();
        marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);
        
        
        Teacher teacher = new Teacher("teacher_001");
        
        Student s1 = new Student("student_001", teacher);
        Student s2 = new Student("student_002", teacher);
        Student s3 = new Student("student_003", teacher);
        
        teacher.students = new ArrayList<>(Arrays.asList(s1, s2, s3));
        
        Classroom classroom = new Classroom();
        classroom.teacher = teacher;
        classroom.students = new ArrayList<>(Arrays.asList(s1, s2, s3));

        try(FileOutputStream out = new FileOutputStream("C:/temp/ref.xml")){
            marshaller.marshal(classroom, out);
        }catch(Exception e){
            e.printStackTrace();
        }
    }
}

 

注意在上面的实现中一共只有一个老师对象和三个学生对象。 序列化的结果为:

<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<classroom>
    <students>
        <student>
            <id>student_001</id>
            <teacher_ref>teacher_001</teacher_ref>
        </student>
        <student>
            <id>student_002</id>
            <teacher_ref>teacher_001</teacher_ref>
        </student>
        <student>
            <id>student_003</id>
            <teacher_ref>teacher_001</teacher_ref>
        </student>
    </students>
    <teacher>
        <id>teacher_001</id>
        <students>
            <student_ref>student_001</student_ref>
            <student_ref>student_002</student_ref>
            <student_ref>student_003</student_ref>
        </students>
    </teacher>
</classroom>

 

 在上面的结果中我们可以清楚的看到 每一个学生和老师的完整数据仅仅出现了一次,其他引用都是仅仅引用了各自的id而已。同样我们对这个结果反序列化,也同样会得到唯一一个老师对象和三个学生对象。

public class UnTest {
    public static void main(String[] args) throws JAXBException {
        JAXBContext context = JAXBContext.newInstance(Classroom.class);
        Unmarshaller u = context.createUnmarshaller();
        
        try(FileInputStream in = new FileInputStream("C:/temp/ref.xml")){
            Classroom classroom = (Classroom)u.unmarshal(in);
            
            Teacher teacher = classroom.teacher;
            
            //教室里所有学生的老师对象和 教室自身的老师对象是同一个对象
            classroom.students.forEach(student -> System.out.println(student.teacher == teacher));
            //教室里所有学生同样也是教室里老师的学生
            classroom.students.forEach(student -> System.out.println(teacher.students.contains(student)));
        }catch(Exception e){
            e.printStackTrace();
        }
    }
}

 

posted @ 2017-04-17 19:54  zh1164  阅读(1516)  评论(0编辑  收藏  举报