Jaxb2 实现JavaBean与xml互转

2 篇文章 0 订阅
1 篇文章 0 订阅

参考: https://blog.csdn.net/xlxxcc/article/details/54612560

            https://blog.csdn.net/nxdxmg/article/details/46926521

            http://www.cnblogs.com/hoojo/archive/2011/04/26/2029011.html

            http://www.oracle.com/technetwork/articles/javase/index-140168.html

            https://javaee.github.io/jaxb-v2/doc/user-guide/index.html

目录

一、简介

二、重要概念

案例:

1.准备工作

2、最简单转换

测试代码:

3、类中包含复杂对象的转换

测试案例

4、集合对象的转换(同样适用于Set)

测试案例

最后  优化:



一、简介

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

      Jaxb 2.0是JDK 1.6的组成部分。我们不需要下载第三方jar包 即可做到轻松转换。Jaxb2使用了JDK的新特性,如:Annotation、GenericType等,需要在即将转换的JavaBean中添加annotation注解。

        XML和Java技术被认为是开发访问Web服务的Web服务和应用程序的理想构建块。一种名为Java Architecture for XML Binding(JAXB)的新Java API可以更容易地从用Java编程语言编写的应用程序访问XML文档。

        

二、重要概念

  • JAXBContext类,是应用的入口,用于管理XML/Java绑定信息。
  • Marshaller接口,将Java对象序列化为XML数据。
  • Unmarshaller接口,将XML数据反序列化为Java对象。

 

  • @XmlType,将Java类或枚举类型映射到XML模式类型,该注解用在class类上,常与@XmlRootElement,@XmlAccessorType一起使用。它有三个属性:name、propOrder、namespace,经常使用的只有前两个属性。如:
    @XmlType(name = "basicStruct", propOrder = {"intValue","stringArray","stringValue"
    )
    Public class myType{…}
    
    在使用@XmlType的propOrder属性时,必须列出JavaBean对象中的所有属性(也要在所有属性上加上xml注解),否则会报错
  • @XmlAccessorType用于指定由java对象生成xml文件时对java对象属性的访问方式。常与@XmlRootElement、@XmlType一起使用。它的属性值是XmlAccessType的4个枚举值,分别为:

      XmlAccessType.FIELD:java对象中的所有成员变量;

      XmlAccessType.PROPERTY:java对象中所有通过getter/setter方式访问的成员变量;

      XmlAccessType.PUBLIC_MEMBER:java对象中所有的public访问权限的成员变量和通过getter/setter方式访问的成员变量;

      XmlAccessType.NONE:java对象的所有属性都不映射为xml的元素。

    注意:@XmlAccessorType的默认访问级别是XmlAccessType.PUBLIC_MEMBER。因此,如果java对象中的private成员变量设置了public权限的getter/setter方法,就不要在private变量上使用@XmlElement和@XmlAttribute注解,否则在由java对象生成xml时会报同一个属性在java类里存在两次的错误。同理,如果@XmlAccessorType的访问权限为XmlAccessType.NONE,如果在java的成员变量上使用了@XmlElement或@XmlAttribute注解,这些成员变量依然可以映射到xml文件。

  • @XmlAccessorOrder,控制JAXB 绑定类中属性和字段的排序。它有两个属性值:

      AccessorOrder.ALPHABETICAL:对生成的xml元素按字母顺序排序;                                                                                 XmlAccessOrder.UNDEFINED:不排序。

  • @XmlJavaTypeAdapter,使用定制的适配器(即扩展抽象类XmlAdapter并覆盖marshal()和unmarshal()方法),以序列化Java类为XML。常用在转换比较复杂的对象时,如map类型或者格式化日期等。使用此注解时,需要自己写一个adapter类继承XmlAdapter抽象类,并实现里面的方法。

      @XmlJavaTypeAdapter(value=xxx.class),value为自己定义的adapter类

    XmlAdapter如下:

  • public abstract class XmlAdapter<ValueType,BoundType> {
        // Do-nothing constructor for the derived classes.
        protected XmlAdapter() {}
        // Convert a value type to a bound type.
        public abstract BoundType unmarshal(ValueType v);
        // Convert a bound type to a value type.
        public abstract ValueType marshal(BoundType v);
     }
    

     

  • @XmlElementWrapper ,对于数组或集合(即包含多个元素的成员变量),生成一个包装该数组或集合的XML元素(称为包装器)。
  •   @XmlTransient  用于标示在由java对象映射xml时,忽略此属性。即,在生成的xml文件中不出现此元素。
  • @XmlRootElement,将Java类或枚举类型映射到XML元素。类级别的注解,对应的是xml文件中的根节点。常与 @XmlType 和 @XmlAccessorType一起使用。如:
  • @XmlType
    @XmlAccessorType(XmlAccessType.FIELD)
    @XmlRootElement
    public class Address {}
    

     

  • @XmlElement,将Java类的一个属性映射到与属性同名的一个XML元素。该注解用在java类的属性上,用于将属性映射为xml的子节点。可通过在后面配置name属性值来改变java属性在xml文件中的名称。如:
  • @XmlElement(name="Address")  
    private String yourAddress;

     此处配置后,在xml文件中的节点为< Address></ Address>,而不是< yourAddress/>。 

  • @XmlAttribute,将Java类的一个属性映射到与属性同名的一个XML属性。用于把java对象的属性映射为xml的属性,并可通过name属性为生成的xml属性指定别名。如:
  • @XmlAttribute(name="Country")
    private String state;

     

    @XmlAttribute(name = "type")
    private String type = "REQUEST";
    
    @XmlAttribute(name = "version")
    private String version = "1.0";

案例:

1.准备工作

package utils;  
  
import java.io.StringReader;  
import java.io.StringWriter;  
  
import javax.xml.bind.JAXBContext;  
import javax.xml.bind.Marshaller;  
import javax.xml.bind.Unmarshaller;  
  
/** 
 * Jaxb2工具类 
 * @author      zhuc 
 * @create      2013-3-29 下午2:40:14 
 */  
public class JaxbUtil {  
  
    /** 
     * JavaBean转换成xml 
     * 默认编码UTF-8 
     * @param obj 
     * @param writer 
     * @return  
     */  
    public static String convertToXml(Object obj) {  
        return convertToXml(obj, "UTF-8");  
    }  
  
    /** 
     * JavaBean转换成xml 
     * @param obj 
     * @param encoding  
     * @return  
     */  
    public static String convertToXml(Object obj, String encoding) {  
        String result = null;  
        try {  
            JAXBContext context = JAXBContext.newInstance(obj.getClass());  
            Marshaller marshaller = context.createMarshaller();  
            marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);  
            marshaller.setProperty(Marshaller.JAXB_ENCODING, encoding);  
  
            StringWriter writer = new StringWriter();  
            marshaller.marshal(obj, writer);  
            result = writer.toString();  
        } catch (Exception e) {  
            e.printStackTrace();  
        }  
  
        return result;  
    }  
  
    /** 
     * xml转换成JavaBean 
     * @param xml 
     * @param c 
     * @return 
     */  
    @SuppressWarnings("unchecked")  
    public static <T> T converyToJavaBean(String xml, Class<T> c) {  
        T t = null;  
        try {  
            JAXBContext context = JAXBContext.newInstance(c);  
            Unmarshaller unmarshaller = context.createUnmarshaller();  
            t = (T) unmarshaller.unmarshal(new StringReader(xml));  
        } catch (Exception e) {  
            e.printStackTrace();  
        }  
  
        return t;  
    }  
}  

  非常简单易懂,需要注意的是:

marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);  
marshaller.setProperty(Marshaller.JAXB_ENCODING, encoding);  

Marshaller.JAXB_FORMATTED_OUTPUT 决定是否在转换成xml时同时进行格式化(即按标签自动换行,否则即是一行的xml)

Marshaller.JAXB_ENCODING   xml的编码方式。

marshaller.setProperty(Marshaller.JAXB_FRAGMENT, fragment); // 是否省略xm头声明信息

另外,Marshaller 还有其他Property可以设置,可以去查阅api。

2、最简单转换

package t1;  
  
import java.util.Date;  
  
import javax.xml.bind.annotation.XmlAccessType;  
import javax.xml.bind.annotation.XmlAccessorType;  
import javax.xml.bind.annotation.XmlAttribute;  
import javax.xml.bind.annotation.XmlElement;  
import javax.xml.bind.annotation.XmlRootElement;  
import javax.xml.bind.annotation.XmlType;  
  
/** 
 * @author      zhuc 
 * @create      2013-3-29 下午2:49:48 
 */  
@XmlAccessorType(XmlAccessType.FIELD)  
@XmlRootElement  
@XmlType(name = "book", propOrder = { "author", "calendar", "price", "id" })  
public class Book {  
  
    @XmlElement(required = true)  
    private String author;  
  
    @XmlElement(name = "price_1", required = true)  
    private float price;  
  
    @XmlElement  
    private Date calendar;  
  
    @XmlAttribute  
    private Integer id;  
  
    /** 
     * @return the author 
     */  
    public String getAuthor() {  
        return author;  
    }  
  
    /** 
     * @return the price 
     */  
    public float getPrice() {  
        return price;  
    }  
  
    /** 
     * @return the calendar 
     */  
    public Date getCalendar() {  
        return calendar;  
    }  
  
    /** 
     * @return the id 
     */  
    public Integer getId() {  
        return id;  
    }  
  
    /** 
     * @param author the author to set 
     */  
    public void setAuthor(String author) {  
        this.author = author;  
    }  
  
    /** 
     * @param price the price to set 
     */  
    public void setPrice(float price) {  
        this.price = price;  
    }  
  
    /** 
     * @param calendar the calendar to set 
     */  
    public void setCalendar(Date calendar) {  
        this.calendar = calendar;  
    }  
  
    /** 
     * @param id the id to set 
     */  
    public void setId(Integer id) {  
        this.id = id;  
    }  
  
    /* (non-Javadoc) 
     * @see java.lang.Object#toString() 
     */  
    @Override  
    public String toString() {  
        return "Book [author=" + author + ", price=" + price + ", calendar=" + calendar + ", id=" + id + "]";  
    }  
  
}  

测试代码:

package t1;  
  
import java.util.Date;  
  
import javax.xml.bind.JAXBException;  
  
import org.junit.Test;  
  
import utils.JaxbUtil;  
  
/** 
 * @author      zhuc 
 * @create      2013-3-29 下午2:50:00 
 */  
public class JaxbTest1 {  
  
    /** 
     * @throws JAXBException 
     */  
    @Test  
    public void showMarshaller()  {  
        Book book = new Book();  
        book.setId(100);  
        book.setAuthor("James");  
        book.setCalendar(new Date());  
        book.setPrice(23.45f);   //默认是0.0  
          
        String str = JaxbUtil.convertToXml(book);  
        System.out.println(str);  
    }  
  
    /** 
     * @throws JAXBException 
     */  
    @Test  
    public void showUnMarshaller() {  
        String str = "<?xml version=\"1.0\" encoding=\"UTF-8\" standalone=\"yes\"?>" +  
            "<book id=\"100\">" +  
            "    <author>James</author>" +  
             "   <calendar>2013-03-29T09:25:56.004+08:00</calendar>" +  
              "  <price_1>23.45</price_1>" +  
            "</book>";  
          
        Book book = JaxbUtil.converyToJavaBean(str, Book.class);  
        System.out.println(book);  
    }  
}  

输出结果分别为:

<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<book id="100">
    <author>James</author>
    <calendar>2013-03-29T14:50:58.974+08:00</calendar>
    <price_1>23.45</price_1>
</book>

Book [author=James, price=23.45, calendar=Fri Mar 29 09:25:56 CST 2013, id=100]

3、类中包含复杂对象的转换

package t2;  
  
import javax.xml.bind.annotation.XmlAccessType;  
import javax.xml.bind.annotation.XmlAccessorType;  
import javax.xml.bind.annotation.XmlAttribute;  
import javax.xml.bind.annotation.XmlElement;  
import javax.xml.bind.annotation.XmlRootElement;  
import javax.xml.bind.annotation.XmlType;  
  
/** 
 * @author      zhuc 
 * @create      2013-3-29 下午2:51:44 
 */  
@XmlAccessorType(XmlAccessType.FIELD)  
@XmlRootElement(name = "student")  
@XmlType(propOrder = {})  
public class Student {  
  
    @XmlAttribute  
    private Integer id;  
  
    @XmlElement  
    private String name;  
  
    @XmlElement(name = "role")  
    private Role role;  
  
    /** 
     * @return the id 
     */  
    public Integer getId() {  
        return id;  
    }  
  
    /** 
     * @return the name 
     */  
    public String getName() {  
        return name;  
    }  
  
    /** 
     * @return the role 
     */  
    public Role getRole() {  
        return role;  
    }  
  
    /** 
     * @param id the id to set 
     */  
    public void setId(Integer id) {  
        this.id = id;  
    }  
  
    /** 
     * @param name the name to set 
     */  
    public void setName(String name) {  
        this.name = name;  
    }  
  
    /** 
     * @param role the role to set 
     */  
    public void setRole(Role role) {  
        this.role = role;  
    }  
  
    /* (non-Javadoc) 
     * @see java.lang.Object#toString() 
     */  
    @Override  
    public String toString() {  
        return "Student [id=" + id + ", name=" + name + ", role=" + role + "]";  
    }  
  
}  
package t2;  
  
import javax.xml.bind.annotation.XmlAccessType;  
import javax.xml.bind.annotation.XmlAccessorType;  
import javax.xml.bind.annotation.XmlElement;  
import javax.xml.bind.annotation.XmlType;  
  
/** 
 * @author      zhuc 
 * @create      2013-3-29 下午2:51:52 
 */  
@XmlAccessorType(XmlAccessType.FIELD)  
@XmlType(propOrder = { "name", "desc" })  
public class Role {  
  
    @XmlElement  
    private String name;  
  
    @XmlElement  
    private String desc;  
  
    /** 
     * @return the name 
     */  
    public String getName() {  
        return name;  
    }  
  
    /** 
     * @return the desc 
     */  
    public String getDesc() {  
        return desc;  
    }  
  
    /** 
     * @param name the name to set 
     */  
    public void setName(String name) {  
        this.name = name;  
    }  
  
    /** 
     * @param desc the desc to set 
     */  
    public void setDesc(String desc) {  
        this.desc = desc;  
    }  
  
    /* (non-Javadoc) 
     * @see java.lang.Object#toString() 
     */  
    @Override  
    public String toString() {  
        return "Role [name=" + name + ", desc=" + desc + "]";  
    }  
  
}  
 

测试案例

package t2;  
  
import org.junit.Test;  
  
import utils.JaxbUtil;  
  
/** 
 * @author      zhuc 
 * @create      2013-3-29 下午2:52:00 
 */  
public class JaxbTest2 {  
  
    @Test  
    public void showMarshaller() {  
  
        Student student = new Student();  
        student.setId(12);  
        student.setName("test");  
  
        Role role = new Role();  
        role.setDesc("管理");  
        role.setName("班长");  
  
        student.setRole(role);  
  
        String str = JaxbUtil.convertToXml(student);  
        System.out.println(str);  
    }  
  
    @Test  
    public void showUnMarshaller() {  
        String str = "<?xml version=\"1.0\" encoding=\"UTF-8\" standalone=\"yes\"?>"+  
            "<student id=\"12\">"+  
            "    <name>test</name>"+  
             "   <role>"+  
              "      <name>班长</name>"+  
               "     <desc>管理</desc>"+  
                "</role>"+  
            "</student>";  
        Student student = JaxbUtil.converyToJavaBean(str, Student.class);  
        System.out.println(student);  
    }  
      

输出结果分别为:

<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<student id="12">
    <name>test</name>
    <role>
        <name>班长</name>
        <desc>管理</desc>
    </role>
</student>

Student [id=12, name=test, role=Role [name=班长, desc=管理]]

4、集合对象的转换(同样适用于Set)

package t3;  
  
import java.util.List;  
  
import javax.xml.bind.annotation.XmlAccessType;  
import javax.xml.bind.annotation.XmlAccessorType;  
import javax.xml.bind.annotation.XmlElement;  
import javax.xml.bind.annotation.XmlElementWrapper;  
import javax.xml.bind.annotation.XmlRootElement;  
import javax.xml.bind.annotation.XmlType;  
  
/** 
 * @author      zhuc 
 * @create      2013-3-29 下午2:55:56 
 */  
@XmlAccessorType(XmlAccessType.FIELD)  
@XmlRootElement(name = "country")  
@XmlType(propOrder = { "name", "provinceList" })  
public class Country {  
  
    @XmlElement(name = "country_name")  
    private String name;  
  
    @XmlElementWrapper(name = "provinces")  
    @XmlElement(name = "province")  
    private List<Province> provinceList;  
  
    /** 
     * @return the name 
     */  
    public String getName() {  
        return name;  
    }  
  
    /** 
     * @return the provinceList 
     */  
    public List<Province> getProvinceList() {  
        return provinceList;  
    }  
  
    /** 
     * @param name the name to set 
     */  
    public void setName(String name) {  
        this.name = name;  
    }  
  
    /** 
     * @param provinceList the provinceList to set 
     */  
    public void setProvinceList(List<Province> provinceList) {  
        this.provinceList = provinceList;  
    }  
  
    /* (non-Javadoc) 
     * @see java.lang.Object#toString() 
     */  
    @Override  
    public String toString() {  
        return "Country [name=" + name + ", provinceList=" + provinceList + "]";  
    }  
  
}  
package t3;  
  
import javax.xml.bind.annotation.XmlAccessType;  
import javax.xml.bind.annotation.XmlAccessorType;  
import javax.xml.bind.annotation.XmlElement;  
import javax.xml.bind.annotation.XmlType;  
  
/** 
 * @author      zhuc 
 * @create      2013-3-29 下午2:56:03 
 */  
@XmlAccessorType(XmlAccessType.FIELD)  
@XmlType(propOrder = { "name", "provCity" })  
public class Province {  
  
    @XmlElement(name = "province_name")  
    private String name;  
  
    @XmlElement(name = "prov_city")  
    private String provCity;  
  
    /** 
     * @return the provCity 
     */  
    public String getProvCity() {  
        return provCity;  
    }  
  
    /** 
     * @param provCity the provCity to set 
     */  
    public void setProvCity(String provCity) {  
        this.provCity = provCity;  
    }  
  
    /** 
     * @return the name 
     */  
    public String getName() {  
        return name;  
    }  
  
    /** 
     * @param name the name to set 
     */  
    public void setName(String name) {  
        this.name = name;  
    }  
  
    /* (non-Javadoc) 
     * @see java.lang.Object#toString() 
     */  
    @Override  
    public String toString() {  
        return "Province [name=" + name + ", provCity=" + provCity + "]";  
    }  
  
}  

测试案例

package t3;  
  
import java.util.ArrayList;  
import java.util.List;  
  
import org.junit.Test;  
  
import utils.JaxbUtil;  
  
/** 
 * @author      zhuc 
 * @create      2013-3-29 下午2:56:11 
 */  
public class JaxbTest3 {  
  
    /** 
     * @throws JAXBException 
     */  
    @Test  
    public void showMarshaller() {  
        Country country = new Country();  
        country.setName("中国");  
  
        List<Province> list = new ArrayList<Province>();  
        Province province = new Province();  
        province.setName("江苏省");  
        province.setProvCity("南京市");  
        Province province2 = new Province();  
        province2.setName("浙江省");  
        province2.setProvCity("杭州市");  
        list.add(province);  
        list.add(province2);  
  
        country.setProvinceList(list);  
  
        String str = JaxbUtil.convertToXml(country);  
        System.out.println(str);  
    }  
  
    /** 
     *  
     */  
    @Test  
    public void showUnMarshaller() {  
        String str = "<?xml version=\"1.0\" encoding=\"UTF-8\" standalone=\"yes\"?>"+  
            "<country>"+  
            "    <country_name>中国</country_name>"+  
            "    <provinces>"+  
            "        <province>"+  
            "            <province_name>江苏省</province_name>"+  
             "           <prov_city>南京市</prov_city>"+  
            "        </province>"+  
             "       <province>"+  
             "           <province_name>浙江省</province_name>"+  
             "           <prov_city>杭州市</prov_city>"+  
             "       </province>"+  
            "    </provinces>"+  
            "</country>";  
        Country country = JaxbUtil.converyToJavaBean(str, Country.class);  
        System.out.println(country);  
    }  
      
}  

输出结果分别为:

<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<country>
    <country_name>中国</country_name>
    <provinces>
        <province>
            <province_name>江苏省</province_name>
            <prov_city>南京市</prov_city>
        </province>
        <province>
            <province_name>浙江省</province_name>
            <prov_city>杭州市</prov_city>
        </province>
    </provinces>
</country>

Country [name=中国, provinceList=[Province [name=江苏省, provCity=南京市], Province [name=浙江省, provCity=杭州市]]]

最后  优化:

     经测试,改两个方法,当调用次数达到一定量的时候,耗时从3ms到33ms,性能急剧下降。经分析发现,是由于JAXBContext.newInstance()方法导致的下降。 
  方法改造如下:

       

static Map<String, JAXBContext> jaxbContextMap = new HashMap<String, JAXBContext>();

    /**
     * java实体类转xml
     */
    public static String toXML(Object obj, String encode, boolean format, boolean fragment) {
        try {
            JAXBContext jaxbContext = jaxbContextMap.get(obj.getClass().getName());
            if(jaxbContext == null){
                // 如果每次都调用JAXBContext.newInstance方法,会导致性能急剧下降
                jaxbContext = JAXBContext.newInstance(obj.getClass());
                jaxbContextMap.put(obj.getClass().getName(), jaxbContext);
            }

            Marshaller marshaller = jaxbContext.createMarshaller();
            marshaller.setProperty(Marshaller.JAXB_ENCODING, encode);           // 编码格式
            marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, format);   // 是否格式化生成的xml串
            marshaller.setProperty(Marshaller.JAXB_FRAGMENT, fragment);         // 是否省略xm头声明信息
            StringWriter writer = new StringWriter();
            marshaller.marshal(obj, writer);
            return writer.toString();
        } catch (Exception e) {
            throw new RuntimeException(e);
        }
    }

    /**
     * xml转java实体类
     */
    @SuppressWarnings("unchecked")
    public static <T> T fromXML(String xml, Class<T> valueType) {
        try {
            JAXBContext jaxbContext = jaxbContextMap.get(valueType.getName());
            if(jaxbContext == null){
                jaxbContext = JAXBContext.newInstance(valueType);
                jaxbContextMap.put(valueType.getName(), jaxbContext);
            }
            Unmarshaller unmarshaller = jaxbContext.createUnmarshaller();
            return (T) unmarshaller.unmarshal(new StringReader(xml));
        } catch (Exception e) {
            throw new RuntimeException(e.getMessage());
        }
    }

 

  • 2
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值