分享一些简化定义和转换Java Bean的一些小技巧

举报
码农小胖哥 发表于 2022/04/01 00:29:26 2022/04/01
【摘要】 今天在封装第三方应用的开放接口,写了很多返回值的类,这些类很多都是结构相似只是个别字段名称不一样。为了单独的字段就要复制一个改改不胜其烦,而且起名是最头疼的事情。就像下面这两个: @EqualsAndHashCode(callSuper = true)@Datapublic class&nbsp...

今天在封装第三方应用的开放接口,写了很多返回值的类,这些类很多都是结构相似只是个别字段名称不一样。为了单独的字段就要复制一个改改不胜其烦,而且起名是最头疼的事情。就像下面这两个:


   
  1. @EqualsAndHashCode(callSuper = true)
  2. @Data
  3. public class SimpleUserResponse extends WeComResponse {
  4.     private List<SimpleUser> userlist;
  5. }
  6. @EqualsAndHashCode(callSuper = true)
  7. @Data
  8. public class UserDetailResponse extends WeComResponse {
  9.     private List<UserDetail> userlist;
  10. }

是不是差不多?于是就利用泛型将它们合并了:


   
  1. @EqualsAndHashCode(callSuper = true)
  2. @Data
  3. public class UserResponse<T> extends WeComResponse {
  4.     private List<T> userlist;
  5. }

这样通过UserResponse<SimpleUser>UserResponse<UserDetail>就能定义他们了,简化了不少代码。不过没多久又来了一个类:


   
  1. @EqualsAndHashCode(callSuper = true)
  2. @Data
  3. public class QrCodeResponse extends WeComResponse {
  4.     private String qrcode;
  5. }

这个结构其实也差不多啊,如果把UserResponse<T>进一步改造成:


   
  1. @EqualsAndHashCode(callSuper = true)
  2. @Data
  3. public class OjbectResponse<T> extends WeComResponse {
  4.     private T userlist;
  5. }

似乎OjbectResponse<String>就等同于QrCodeResponse。不过单纯这样搞是不行的,细心的同学会发现它们的属性名称不一样,一个是qrcode;一个是userlist如果能起个别名就好了!我似乎有了一个解决方案。

如果是类型转换

Bean类型转换的话,用Mapstruct就能解决这个问题,最终我们把属性名称定义为data


   
  1.     @Mapping(target = "data", source = "qrcode")
  2.     @Mapping(target = "data", source = "userlist")

通过上面的两个注解映射编写两个转换接口就解决了。关于Mapstruct可以看我相关的讲解文章

如果是反序列化

Jackson提供了一个别名注解@JsonAlias,可以让字段属性名称接受更多的别名。就像这样:


   
  1. @EqualsAndHashCode(callSuper = true)
  2. @Data
  3. public class OjbectResponse<T> extends WeComResponse {
  4.     @JsonAlias({"qrcode","userlist"})
  5.     private T data;
  6. }

那么下面这个json能映射到OjbectResponse<String>中:


   
  1. {
  2.     "qrcode":"https://felord.cn/myqr.png"
  3. }

这个会映射到OjbectResponse<List<UserDetail>>中:


   
  1. {
  2.  "userlist":[{"username":"felord.cn"},{"username":"felordcn"},{"username":"felord"}]
  3. }

到这里可能大家会有疑问:Jackson是如何处理泛型问题的?

如何获取泛型的Class类型

通过直接手段是无法获取泛型的Class类型的,不过我们可以获取到泛型的抽象定义java.lang.reflect.ParameterizedType,直接使用ParameterizedType会不太方便。所以在Jackson中可以通过TypeReference<T>来处理泛型问题。如果我们需要反序列化OjbectResponse<String>我们可以:


   
  1. ObjectMapper objectMapper = new ObjectMapper();
  2. String json = "{\"qrcode\":\"https://felord.cn/myqr.png\"}";
  3. OjbectResponse<String> obj = objectMapper.readValue(json,new TypeReference<OjbectResponse<String>>(){});

其实对应的Spring也提供了类似的工具类org.springframework.core.ParameterizedTypeReference<T>,特别是如果你使用了RestTemplate来请求第三方的时候会用到这个泛型处理工具。

Java17 新特性确定,Java之父:终于可以和一个长达25年的漏洞说再见了

2021-06-16

这个类库可以帮助你理解Java中的函数式编程

2021-06-15

文章来源: felord.blog.csdn.net,作者:码农小胖哥,版权归原作者所有,如需转载,请联系作者。

原文链接:felord.blog.csdn.net/article/details/118036399

【版权声明】本文为华为云社区用户转载文章,如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。