前言:
1、两套标准:Jax-ws(web service)和 Jax-rs(restful)。
2、在JAVA EE 5\6中,jaxb可以很方便的与jax-rs、jax-ws集成,极大的简化了web service接口的开发工作量。jaxb负责xml与javaBean的映射。
3、cxf等架构的使用,ws中不需要手动写映射,而restfu则需要手动书写注解。(也可以使用JAXB的XJC工具,通过定义schema的方式实现Java对象与XML的绑定)。
4、jaxb除了在ws与restful中广泛使用,也可以作为解析xml的一种日常技术。
一、web service中的jaxb
1、在使用cxf的客户端时,一般使用 wsdl2java 命令生成,所以不需要自己写jaxb的代码。如:
@XmlAccessorType(XmlAccessType.FIELD) @XmlType(name = "cat",propOrder = { "color","id","name" }) public class Cat { protected String color; protected Integer id; protected String name; //...... }
2、其实这边映射的就是 soap消息,与访问的 wsdl 具有一致性。
二、jaxb详解。
(一)实例
1、javaBean实例:
package cn.rest.bean; import java.util.List; import javax.xml.bind.annotation.XmlAttribute; import javax.xml.bind.annotation.XmlElement; import javax.xml.bind.annotation.XmlElementWrapper; import javax.xml.bind.annotation.XmlRootElement; /** * * UserBean.java * * @title User的传输数据类 * @description * @author SAM-SHO * @Date 2014-11-25 */ @XmlRootElement(name = "USER") //@XmlAccessorType(XmlAccessType.FIELD ) //写这个注解,就不需要@XmlElement注解,一般和XmlType一起使用,指定名称。 //@XmlType(propOrder = { "name","age","userAddress","phoneList"}) public class UserBean { private String name; private String key; private String age; private UserAddress userAddress;//地址 private List<UserPhone> phoneList ;//手机 @XmlElement(name="NAME") public String getName() { return name; } public void setName(String name) { this.name = name; } @XmlElement(name = "AGE") public String getAge() { return age; } public void setAge(String age) { this.age = age; } @XmlElement(name = "UserAddress") public UserAddress getUserAddress() { return userAddress; } public void setUserAddress(UserAddress userAddress) { this.userAddress = userAddress; } @XmlElementWrapper(name = "PhoneList")//标注集合 @XmlElement(name = "UserPhone") public List<UserPhone> getPhoneList() { return phoneList; } public void setPhoneList(List<UserPhone> phoneList) { this.phoneList = phoneList; } @XmlAttribute(name="key")//属性 public String getKey() { return key; } public void setKey(String key) { this.key = key; } }
package cn.rest.bean; public class UserPhone { private String type;//电话号码类型 private String num;//电话号码 public String getType() { return type; } public void setType(String type) { this.type = type; } public String getNum() { return num; } public void setNum(String num) { this.num = num; } }
package cn.rest.bean; import javax.xml.bind.annotation.XmlElement; public class UserAddress { private String homeAddress;//家庭地址 private String workAddress;//公司地址 @XmlElement(name = "HomeAddress") public String getHomeAddress() { return homeAddress; } public void setHomeAddress(String homeAddress) { this.homeAddress = homeAddress; } @XmlElement(name = "WorkAddress") public String getWorkAddress() { return workAddress; } public void setWorkAddress(String workAddress) { this.workAddress = workAddress; } }
2、解析工具。
1)javaBean 映射成 xml,使用Marshaller 。
/** * Objext 转成 XML * * @param object */ public static void Object2Xml(Object object) { // FileWriter writer = null; try { JAXBContext context = JAXBContext.newInstance(object.getClass()); Marshaller marshal = context.createMarshaller(); marshal.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT,true); // 格式化输出 marshal.setProperty(Marshaller.JAXB_ENCODING,"utf-8");// 编码格式,默认为utf-8 marshal.setProperty(Marshaller.JAXB_FRAGMENT,false);// 是否省略xml头信息 marshal.marshal(object,System.out); // writer = new FileWriter("shop.xml"); // marshal.marshal(object,writer); } catch (Exception e) { e.printStackTrace(); } }
2)xml 映射成 javaBean,使用 Unmarshaller。
/** * xml 转成特定的Bean * * @param xml * @return */ public static UserBean parseXml2UserBean(String xml) { try { JAXBContext context = JAXBContext.newInstance(UserBean.class); InputSource is = new InputSource(); StringReader xmlStr = new StringReader(xml); is.setCharacterStream(xmlStr); Unmarshaller unmarshaller = context.createUnmarshaller(); UserBean user = (UserBean) unmarshaller.unmarshal(is); return user; } catch (JAXBException e) { e.printStackTrace(); return null; } }
3)测试:
UserBean tUserBean = new UserBean(); tUserBean.setName("SAM-SHO"); tUserBean.setAge("27"); tUserBean.setKey("属性111"); UserAddress tUserAddress = new UserAddress(); tUserAddress.setWorkAddress("苏州园区"); tUserAddress.setHomeAddress("苏州高新区"); tUserBean.setUserAddress(tUserAddress); List<UserPhone> phoneList = new ArrayList<UserPhone>(); UserPhone tUserPhone = new UserPhone(); tUserPhone.setType("移动"); tUserPhone.setNum("13612345678"); phoneList.add(tUserPhone); tUserPhone = new UserPhone(); tUserPhone.setType("联通"); tUserPhone.setNum("13798765432"); phoneList.add(tUserPhone); tUserBean.setPhoneList(phoneList); ObjectAndXmlHandle.Object2Xml(tUserBean);
【输出】
<?xml version="1.0" encoding="utf-8" standalone="yes"?> <USER key="属性111"> <AGE>27</AGE> <NAME>SAM-SHO</NAME> <PhoneList> <UserPhone> <num>13612345678</num> <type>移动</type> </UserPhone> <UserPhone> <num>13798765432</num> <type>联通</type> </UserPhone> </PhoneList> <UserAddress> <HomeAddress>苏州高新区</HomeAddress> <WorkAddress>苏州园区</WorkAddress> </UserAddress> </USER>
(二)jaxb 注解详解:
1、@XmlRootElement 将一个Java类映射为一段XML的根节点。参数:
name 定义这个根节点的名称
namespace 定义这个根节点命名空间
2、@XmlAccessorType 定义映射这个类中的何种类型需要映射到XML。可接收四个参数,分别是:
XmlAccessType.FIELD:映射这个类中的所有字段到XML
XmlAccessType.PUBLIC_MEMBER:将这个类中的所有public的field或property同时映射到XML(默认)
XmlAccessType.NONE:不映射
3、@XmlElement 指定一个字段或get/set方法映射到XML的节点。如,当一个类的XmlAccessorType 被标注为
PROPERTY时,在某一个没有get/set方法的字段上标注此注解,即可将该字段映射到XML。参数:
defaultValue 指定节点默认值
name 指定节点名称
namespace 指定节点命名空间
required 是否必须(默认为false)
nillable 该字段是否包含 nillable="true" 属性(默认为false)
type 定义该字段或属性的关联类型
4、@XmlAttribute 指定一个字段或get/set方法映射到XML的属性。参数:
namespace 指定属性命名空间
required 是否必须(默认为false)
5、@XmlTransient 定义某一字段或属性不需要被映射为XML。如,当一个类的XmlAccessorType 被标注为PROPERTY时,在某一get/set方法的字段上标注此注解,那么该属性则不会被映射。
6、@XmlType 定义映射的一些相关规则。参数:
propOrder 指定映射XML时的节点顺序
actoryClass 指定UnMarshal时生成映射类实例所需的工厂类,默认为这个类本身
factoryMethod 指定工厂类的工厂方法
name 定义XML Schema中type的名称
7、@XmlElementWrapper 为数组元素或集合元素定义一个父节点。如,类中有一元素为List items,若不加此注解,该元素将被映射为namespace 指定Schema中的命名空间
<items>...</items> <items>...</items>
这种形式,此注解可将这个元素进行包装,如:
@XmlElementWrapper(name="items") @XmlElement(name="item") public List items;
将会生成这样的XML样式:
<items> <item>...</item> <item>...</item> </items>
8、@XmlJavaTypeAdapter 自定义某一字段或属性映射到XML的适配器。如,类中包含一个接口,我们可以定义一个适配器(继承自 javax.xml.bind.annotation.adapters.XmlAdapter类),指定这个接口如何映射到XML。 9、@XmlSchema 配置整个包的namespace,这个注解需放在package-info.java文件中。