前言:
  最近的工作内容跟银行有些交互, 对方提供的数据格式采用xml(不是预期的json/protobuf). 为了开发方便, 需要借助jaxb来实现xml和java对象之间的映射. 它还是有点像jackson, 通过简单的注解配置, 就能轻松实现json和java对象的互转. 不过笔者在java类中引入泛型时, 还是踩了不少jaxb的坑, 这边做下笔记.

实现的目标:
  交互的数据格式和协议遵循通用的设计, 由header和body构成.
  请求的数据格式如下:

<?xml version="1.0" encoding="UTF-8" ?>
<root>
<!-- 请求头 -->
<header></header>
<request>
<!-- 具体的请求参数, 根据接口而定 -->
</request>
</root>

  响应的数据格式如下:

<?xml version="1.0" encoding="UTF-8" ?>
<root>
<!-- 响应头 -->
<header></header>
<response>
<!-- 具体的响应结果, 根据接口而定 -->
</response>
</root>

  header信息头相对固定, 而具体的request/response取决于具体的业务接口, 在进行对象映射中, 我们也是针对body消息体进行泛型化.

请求类抽象和测试代码:
  针对请求的数据格式, 我们可以轻易的设计如下类结构:

// *) 请求类(模板)
@Getter
@Setter
@ToString
public class Req<T> {
private String header;
private T value;
} // *) 具体的实体请求
@Getter
@Setter
@ToString
@AllArgsConstructor
@NoArgsConstructor
public class EchoBody {
private String key;
}

  注: 这边的注解Getter/Setter/ToString等皆是lombok的注解.
  测试代码如下:

    public static void main(String[] args) {

        Req<EchoBody> req = new Req<EchoBody>();
req.setHeader("header");
req.setValue(new EchoBody("key")); try {
StringWriter sw = new StringWriter();
JAXBContext context = JAXBContext.newInstance(req.getClass());
Marshaller marshaller = context.createMarshaller();
marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);
marshaller.marshal(req, sw);
System.out.println(sw.toString());
} catch (JAXBException e) {
e.printStackTrace();
} }

  注: 该代码主要测试对象到xml的转换是否顺利.

演进和迭代:
  先来看第一版本, 引入jaxb注解, 同时省略lombok注解.

    @XmlRootElement(name="root")
@XmlAccessorType(XmlAccessType.FIELD)
public class Req<T> {
@XmlElement(name="header",required = true)
private String header;
@XmlElement(name="request", required = true)
private T value;
} @XmlRootElement(name="request")
@XmlAccessorType(XmlAccessType.FIELD)
public class EchoBody {
@XmlElement(name="key", required = true)
private String key;
}

  运行测试的结果如下:

javax.xml.bind.MarshalException
- with linked exception:
[com.sun.istack.internal.SAXException2: class com.test.Test$EchoBody以及其任何超类对此上下文都是未知的。
javax.xml.bind.JAXBException: class com.test.Test$EchoBody以及其任何超类对此上下文都是未知的。]
at com.sun.xml.internal.bind.v2.runtime.MarshallerImpl.write(MarshallerImpl.java:311)
at com.sun.xml.internal.bind.v2.runtime.MarshallerImpl.marshal(MarshallerImpl.java:236)
at javax.xml.bind.helpers.AbstractMarshallerImpl.marshal(AbstractMarshallerImpl.java:116)
at com.test.Test.main(Test.java:55)

  来首战遇到一些小挫折, 通过百度得知需要借助@XmlSeeAlso类规避该问题.
  修改代码如下:

    @XmlRootElement(name="root")
@XmlAccessorType(XmlAccessType.FIELD)
@XmlSeeAlso({EchoBody.class})
public class Req<T> {
@XmlElement(name="header",required = true)
private String header;
@XmlElement(name="request", required = true)
private T value;
} @XmlRootElement(name="request")
@XmlAccessorType(XmlAccessType.FIELD)
public class EchoBody {
@XmlElement(name="key", required = true)
private String key;
}

  运行后的输出结果如下:

<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<root>
<header>header</header>
<request xsi:type="echoBody" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
<key>key</key>
</request>
</root>

  看来非常的成功, 但是request标签里包含了xsi:type和xmlns:xsi这些属性, 能否把这些信息去除, 网上查阅得知, 借助@XmlAnyElement(lax = true)来达到目的, 再次修改版本.

    @XmlRootElement(name="root")
@XmlAccessorType(XmlAccessType.FIELD)
@XmlSeeAlso({EchoBody.class})
public class Req<T> {
@XmlElement(name="header",required = true)
private String header;
@XmlAnyElement(lax = true)
private T value;
} @XmlRootElement(name="request")
@XmlAccessorType(XmlAccessType.FIELD)
public class EchoBody {
@XmlElement(name="key", required = true)
private String key;
}

  这次的结果可以称得上完美(perfect):

<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<root>
<header>header</header>
<request>
<key>key</key>
</request>
</root>

  

响应类抽象和测试代码:
  有了请求类的顺利结果, 我们在设计响应类也是有迹可循.
  响应类的代码如下:

@Getter
@Setter
@ToString
public class Res<T> {
private String header;
private T value;
} @Getter
@Setter
@ToString
@AllArgsConstructor
@NoArgsConstructor
public class EchoAck {
private String value;
} @Getter
@Setter
@ToString
@AllArgsConstructor
@NoArgsConstructor
public class HelloAck {
private String key;
}

  注: 这边暂时隐去jaxb的注解, 剩下的都是lombok注解.
  测试用例代码如下:

public static void main(String[] args) {

    String xml = "" +
"<?xml version=\"1.0\" encoding=\"UTF-8\" ?>\n" +
"<root>\n" +
"\t<header>header_val</header>\n" +
"\t<response>\n" +
"\t\t<key>key_val</key>\n" +
"\t</response>\n" +
"</root>";
Res<HelloAck> res = new Res<HelloAck>(); try {
JAXBContext jc = JAXBContext.newInstance(res.getClass());
Unmarshaller unmar = jc.createUnmarshaller();
Res<HelloAck> r = (Res<HelloAck>)unmar.unmarshal(new StringReader(xml));
System.out.println(r);
} catch (JAXBException e) {
e.printStackTrace();
} }

  

演进和迭代:
  添加jaxb注解, 隐去lombok注解, 大致如下:

@XmlRootElement(name="root")
@XmlAccessorType(XmlAccessType.FIELD)
@XmlSeeAlso({HelloAck.class, EchoAck.class})
public class Res<T> {
@XmlElement(name="header",required = true)
private String header;
@XmlAnyElement(lax = true)
private T value;
} @XmlRootElement(name="response")
@XmlAccessorType(XmlAccessType.FIELD)
public class EchoAck {
@XmlElement(name="value", required = true)
private String value;
} @XmlRootElement(name="response")
@XmlAccessorType(XmlAccessType.FIELD)
public class HelloAck {
@XmlElement(name="key", required = true)
private String key;
}

  运行的如下:

Res(header=header_val, value=EchoAck(value=null))

  这边需要的注意的是, 代码中指定反解的类是HelloAck, 但是这边反解的类却是EchoAck. 由此可见, jaxb在xml到对象转换时, 其泛型类的选取存在问题(猜测java泛型在编译时类型被擦去, 反射不能确定具体那个类).
  针对这种情况, 一个好的建议是, 单独引入实体类(wrapper), 网友的做法也是类似, 只是没有给出直接的理由.

@Getter
@Setter
@ToString
@XmlTransient // 抽象基类改为注解XmlTransient, 切记
@XmlAccessorType(XmlAccessType.FIELD)
public abstract class Res<T> {
@XmlElement(name="header",required = true)
private String header;
@XmlAnyElement(lax = true)
private T value;
} @Getter
@Setter
@ToString
@AllArgsConstructor
@NoArgsConstructor
@XmlRootElement(name="response")
@XmlAccessorType(XmlAccessType.FIELD)
public class EchoAck {
@XmlElement(name="value", required = true)
private String value;
} @Getter
@Setter
@ToString
@AllArgsConstructor
@NoArgsConstructor
@XmlRootElement(name="response")
@XmlAccessorType(XmlAccessType.FIELD)
public class HelloAck {
@XmlElement(name="key", required = true)
private String key;
} @Getter
@Setter
@ToString(callSuper = true)
@XmlRootElement(name="root")
@XmlAccessorType(XmlAccessType.FIELD)
@XmlSeeAlso({HelloAck.class})
public class HelloRes extends Res<HelloAck> {
}

  修改测试代码:

public static void main(String[] args) {

    String xml = "" +
"<?xml version=\"1.0\" encoding=\"UTF-8\" ?>\n" +
"<root>\n" +
"\t<header>header_val</header>\n" +
"\t<response>\n" +
"\t\t<key>key_val</key>\n" +
"\t</response>\n" +
"</root>";
HelloRes res = new HelloRes(); try {
JAXBContext jc = JAXBContext.newInstance(HelloRes.class);
Unmarshaller unmar = jc.createUnmarshaller();
HelloRes r = (HelloRes)unmar.unmarshal(new StringReader(xml));
System.out.println(r);
} catch (JAXBException e) {
e.printStackTrace();
} }

  运行结果如下:

HelloRes(super=Res(header=header_val, value=HelloAck(key=key_val)))

  符合预期, 这边的做法就是wrap一个泛型类, 姑且可以理解为在编译前指定类, 避免反射出偏差.

总结:
  总的来说jaxb在涉及泛型时, 还是有一些坑的, 这边总结了一下. 不过总的来说, 知其然不知其所以然, 希翼后面能够对jaxb的底层实现有个深入的了解.

当Jaxb遇到泛型的更多相关文章

  1. Jaxb解析xml准换为javabean

    先说下这个的背景吧,前些日子,有个以前的小同事说刚接触webservice,想解析下xml,记得我学的时候还是dom4j,sax的解析方式,最近看别人的代码用的jaxb的方式,觉得注解起来很简练,所以 ...

  2. 使用JAXB读写xml

    1.注解 在Java模型中的创建与 xml 文件对应的节点和属性需要用注解来表示 @XmlRootElement 作用:将一个Java类映射为一段XML的根节点参数:    name  定义这个根节点 ...

  3. 【转载】在Jersey JAX-RS 处理泛型List等Collection

    在Java中,从1.5开始,我们就可以使用泛型了(generic),这看上去很像C++ Template,但是实际上它们是不同的.在这里我不想过多的描述细节,你可以从Google上搜索一下. 但是,泛 ...

  4. JAXB 专题二(BSP接口实战)

    BSP下单接口 1.xml格式如下 <?xml version="1.0" encoding="utf-8"?> <Request servi ...

  5. JAXB读写xml

    一.注解 在Java模型中的创建与 xml 文件对应的节点和属性需要用注解来表示 @XmlRootElement 作用:将一个Java类映射为一段XML的根节点参数:    name  定义这个根节点 ...

  6. 一起学 Java(三) 集合框架、数据结构、泛型

    一.Java 集合框架 集合框架是一个用来代表和操纵集合的统一架构.所有的集合框架都包含如下内容: 接口:是代表集合的抽象数据类型.接口允许集合独立操纵其代表的细节.在面向对象的语言,接口通常形成一个 ...

  7. .NET面试题系列[8] - 泛型

    “可变性是以一种类型安全的方式,将一个对象作为另一个对象来使用.“ - Jon Skeet .NET面试题系列目录 .NET面试题系列[1] - .NET框架基础知识(1) .NET面试题系列[2] ...

  8. C#4.0泛型的协变,逆变深入剖析

    C#4.0中有一个新特性:协变与逆变.可能很多人在开发过程中不常用到,但是深入的了解他们,肯定是有好处的. 协变和逆变体现在泛型的接口和委托上面,也就是对泛型参数的声明,可以声明为协变,或者逆变.什么 ...

  9. 编写高质量代码:改善Java程序的151个建议(第7章:泛型和反射___建议106~109)

    建议106:动态代理可以使代理模式更加灵活 Java的反射框架提供了动态代理(Dynamic Proxy)机制,允许在运行期对目标类生成代理,避免重复开发.我们知道一个静态代理是通过主题角色(Prox ...

随机推荐

  1. Win32汇编环境搭建教程(MASM32 SDK)

    一.说明 常用的32位汇编编译器有微软的MASM.Borland的TASM和NASM. 编译器 开发者 优点 缺点 MASM 微软 微软自家软件和系统兼容性好:支持invoke/.if等伪指令将汇编变 ...

  2. Java类型中ParameterizedType,GenericArrayType,TypeVariabl,WildcardType详解

    (1). 和反射+泛型有关的接口类型 java.lang.reflect.Type:java语言中所有类型的公共父接口 java.lang.reflect.ParameterizedType java ...

  3. 各种形式的熵函数,KL距离

    自信息量I(x)=-log(p(x)),其他依次类推. 离散变量x的熵H(x)=E(I(x))=-$\sum\limits_{x}{p(x)lnp(x)}$ 连续变量x的微分熵H(x)=E(I(x)) ...

  4. Debug method

    #define DEBUG(format,...) printf("Ray.he file:"__FILE__" func:%s() line:%d, print &qu ...

  5. 一款c语言实现的赛车游戏

    博主学习c语言已经有一段时间了,出于对自己学习检验的目的,自制了一款c语言赛车游戏. 由于本质是检验和尝试,所以并没有注重游戏的界面.下文是开发文档,在博主的github网页可以下载源码,注意本项目使 ...

  6. unity3D打包发布Apk详细步骤

    1.复制android-sdk-windows文件夹到C盘或者D盘或者你可以找到的任意盘任意目录,注意:不能在中文目录下!! 复制完成之后,打开unity,新建一个项目,打开Edit-Preferen ...

  7. OOP⑻

    1.接口: 类 和 对象 对象 is a 类 例子: 小鸟 is a 动物 飞机 is a 交通工具 子弹 is a 武器 卫星 is a 通讯工具 问题? 01. 小鸟 飞机 子弹 卫星 虽然不是一 ...

  8. TTL特殊门电路

    集电极开路(OC)门:主要作用实现线与功能:用做驱动器:实现电平转换 三态输出(TS)门:应用于计算机总线结构,通过分时控制三态门始轮端使得cpu与不同的外设通信:应用于双向传输,实现门电路与总线实现 ...

  9. Java实现将数字转为大写汉字

    public class Int2Big { static String int2big(int src) { final String num[] = {"零", "壹 ...

  10. 3.9 C++多继承

    参考:http://www.weixueyuan.net/view/6366.html 总结: C++中一个派生类中允许有两个及以上的基类,我们称这种情况为多继承 使用多继承可以描述事物之间的组合关系 ...