服务器之家:专注于服务器技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - Java教程 - 一些简化定义和转换Java Bean的一些小技巧

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

2021-06-18 23:36码农小胖哥 Java教程

本文主要分享一些简化定义和转换Java Bean的一些小技巧,有需要的朋友可以参考下

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

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

  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; 

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

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

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

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

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

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

似乎OjbectResponse就等同于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; 

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

  1.     "qrcode":"https://felord.cn/myqr.png" 

这个会映射到OjbectResponse

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

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

如何获取泛型的Class类型

 

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

  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,特别是如果你使用了RestTemplate来请求第三方的时候会用到这个泛型处理工具。

原文链接:https://mp.weixin.qq.com/s/XQT8Qrx9yQdjjKNHGsfePQ

延伸 · 阅读

精彩推荐