成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

SpringBoot 使用轉換器將前端參數轉換為枚舉

開發 架構
最近遇到一個小伙伴問前端枚舉轉換問題,才意識到可以通過轉換器(Converter)自動將前端傳入的字段值使用枚舉接收。

[[421671]]

前言

最近遇到一個小伙伴問前端枚舉轉換問題,才意識到可以通過轉換器(Converter)自動將前端傳入的字段值使用枚舉接收。

我自己搗鼓了一番,現在記錄筆記分享一下!有興趣的小伙伴可以自己嘗試一下!

這里使用的是 MyBatis-Plus 和 SpringBoot 2.3.4.RELEASE

1實現過程

配置轉換器

  1. /** 
  2.  * @author liuzhihang 
  3.  * @date 2021/8/31 16:29 
  4.  */ 
  5. @Configuration 
  6. public class WebConfig implements WebMvcConfigurer { 
  7.  
  8.     @Override 
  9.     public void addFormatters(FormatterRegistry registry) { 
  10.  
  11.         registry.addConverterFactory(new ConverterFactory<Object, BaseEnum>() { 
  12.             @Override 
  13.             public <T extends BaseEnum> Converter<Object, T> getConverter(Class<T> targetType) { 
  14.  
  15.                 T[] enums = targetType.getEnumConstants(); 
  16.  
  17.                 return source -> { 
  18.  
  19.                     for (T e : enums) { 
  20.                         if (e.getCode().equals(source)) { 
  21.                             return e; 
  22.                         } 
  23.                     } 
  24.  
  25.                     throw new IllegalArgumentException("枚舉 Code 不正確"); 
  26.                 }; 
  27.             } 
  28.         }); 
  29.     } 

直接在 WebMvcConfigurer 里實現 addFormatters 方法即可,然后 new 一個 ConverterFactory。

WebMvcConfigurer 相信大家都不陌生,一般添加一些攔截器,通用校驗 token、日志等等都會用到。具體可以參考這篇文章:幾行代碼輕松實現跨系統傳遞 traceId,再也不用擔心對不上日志了!,里面有一些其他的應用。

就這些,很簡單的實現。下面介紹下項目的內容和代碼,方便理解。

項目代碼

  • 請求參數:
  1. POST http://localhost:8818/user/listByStatus 
  2. Content-Type: application/json 
  3.  
  4.   "orderStatus": 1 
  • Controller
  1. /** 
  2.  * @author liuzhihang 
  3.  * @date 2021/8/30 11:08 
  4.  */ 
  5. @Slf4j 
  6. @RestController 
  7. @RequestMapping("/user"
  8. public class UserController { 
  9.  
  10.     @Autowired 
  11.     private OrderService orderService; 
  12.  
  13.     @PostMapping(value = "/listByStatus"
  14.     public ResultVO<UserResponse> listByStatus(@Validated @RequestBody UserRequest request)  { 
  15.  
  16.         log.info("請求參數:{}", request); 
  17.  
  18.         List<TransOrder> orderList = orderService.getByOrderStatus(request.getOrderStatus()); 
  19.  
  20.         UserResponse response = new UserResponse(); 
  21.  
  22.         response.setRecords(orderList); 
  23.  
  24.         log.info("返回參數:{}", response); 
  25.  
  26.         return ResultVO.success(response); 
  27.     } 
  • Entity
  1. @Data 
  2. public class UserRequest { 
  3.  
  4.     private OrderStatusEnum orderStatus; 
  5.     private ViewStatusEnum viewStatus; 
  6.  
  7. @Data 
  8. public class UserResponse { 
  9.  
  10.     private List<TransOrder> records; 
  11.  

Web 傳入 orderStatus 為 1,而后端接收對象是 UserRequest 的 orderStatus 字段是個 OrderStatusEnum 類型的枚舉。

這里就需要自動將數字類型的字段轉換為枚舉字段。這個枚舉會直接通過 MyBatis-Plus 查詢。

為什么要這么用呢?

其實原因很簡單,使用枚舉限制數據庫字段的類型,比如數據庫狀態只有 0、1、2,那就和代碼里的枚舉對應起來。防止傳入其他值。

  • 枚舉
  1. public interface BaseEnum { 
  2.     Object getCode(); 
  1. public enum OrderStatusEnum implements BaseEnum { 
  2.  
  3.     INIT(0, "初始狀態"), 
  4.     SUCCESS(1, "成功"), 
  5.     FAIL(2, "失敗"); 
  6.  
  7.     @EnumValue 
  8.     @JsonValue 
  9.     private final int code; 
  10.  
  11.     private final String desc
  12.  
  13.     OrderStatusEnum(int code, String desc) { 
  14.         this.code = code; 
  15.         this.desc = desc
  16.     } 
  17.  
  18.     @Override 
  19.     public Integer getCode() { 
  20.         return code; 
  21.     } 
  22.  
  23.     public String getDesc() { 
  24.         return desc
  25.     } 

這里先聲明接口 BaseEnum,所有的枚舉都繼承這個接口,并實現 getCode 方法。

@EnumValue:MyBatis-Plus 的枚舉,和數據庫字段映射用的

@JsonValue:返回給前端時,這個枚舉字段序列化時,返回參數只顯示 code。

這樣就可以實現效果,請求參數為數字,接收對象字段為枚舉,返回字段也是 code。

效果

測試結果

測試結果經過驗證,是可以勝任傳入數值和字符串的。

也可以結合異常處理器,返回通用異常。具體怎么用查一查 @ExceptionHandler 就知道了。

具體說明

在 addFormatters 方法中可以看到 registry.addConverterFactory() 接收的是一個 ConverterFactory 對象。

  1. public interface ConverterFactory<S, R> { 
  2.  
  3.  <T extends R> Converter<S, T> getConverter(Class<T> targetType); 
  • S 就是傳入的字段類型(數字,字符串)
  • R 是要轉換為的類型(枚舉)
  • T 繼承了 R,其實就是參數對象中字段的類型

在 ConverterFactory 的 getConverter 方法則需要返回一個實際的轉換器 Converter

  1. @FunctionalInterface 
  2. public interface Converter<S, T> { 
  3.  
  4.  @Nullable 
  5.  T convert(S source); 
  6.  

convert 方法的入參是一個 source,就是要轉換為什么類型的,這里就是數字/字符串,然后返回一個枚舉即可。

注意這里加了 @FunctionalInterface 就意味著這里是可以用 lambda 表達式的。

2優化

一般 WebConfig 中除了實現 addFormatters 方法外,還會實現 addInterceptors 等等,這樣寫難免會很長,所以可以改為下面這種。

  1. @Configuration 
  2. public class WebConfig implements WebMvcConfigurer { 
  3.  
  4.     @Autowired 
  5.     private LogInterceptor logInterceptor; 
  6.  
  7.     @Autowired 
  8.     private AppTokenInterceptor appTokenInterceptor; 
  9.  
  10.  
  11.     @Autowired 
  12.     private EnumConverterFactory enumConverterFactory; 
  13.  
  14.     @Override 
  15.     public void addInterceptors(InterceptorRegistry registry) { 
  16.  
  17.         // 日志 
  18.         registry.addInterceptor(logInterceptor) 
  19.                 .addPathPatterns("/**"); 
  20.  
  21.         // app token校驗 
  22.         registry.addInterceptor(appTokenInterceptor) 
  23.                 .addPathPatterns("/app/**"); 
  24.  
  25.     } 
  26.  
  27.     @Override 
  28.     public void addFormatters(FormatterRegistry registry) { 
  29.          
  30.         // 枚舉轉換 
  31.         registry.addConverterFactory(enumConverterFactory); 
  32.     } 

這種就需要咱們創建 EnumConverterFactory 類并實現 ConverterFactory 接口了,還得注入到 Spring 容器中

  1. @Component 
  2. public class EnumConverterFactory implements ConverterFactory<Object, BaseEnum> { 
  3.  
  4.     @Override 
  5.     public <T extends BaseEnum> Converter<Object, T> getConverter(Class<T> targetType) { 
  6.  
  7.         return new EnumConverter<>(targetType); 
  8.     } 
  9. public class EnumConverter<T extends BaseEnum> implements Converter<Object, T> { 
  10.  
  11.     private final Class<T> targetType; 
  12.  
  13.     public EnumConverter(Class<T> targetType) { 
  14.         this.targetType = targetType; 
  15.     } 
  16.  
  17.     @Override 
  18.     public T convert(Object source) { 
  19.  
  20.         for (T e : targetType.getEnumConstants()) { 
  21.             if (e.getCode().equals(source)) { 
  22.                 return e; 
  23.             } 
  24.         } 
  25.  
  26.         throw new IllegalArgumentException("枚舉 Code 不正確"); 
  27.     } 

3總結

當然這里也有一些其他的優化點,比如可以使用緩存將 Convert 緩存起來。

 

不過我也遇到一個其他的問題,就是我 debug 斷點竟然一直沒有斷到轉換器中,不知道有沒有小伙伴嘗試過?

 

責任編輯:武曉燕 來源: 程序員小航
相關推薦

2010-06-10 14:33:03

協議轉換器

2010-06-10 14:44:33

協議轉換器

2013-06-13 15:10:27

.NET代碼轉換

2010-06-10 14:38:30

協議轉換器

2010-06-10 15:03:13

協議轉換器

2014-05-04 12:51:21

Javascript編譯器

2023-05-05 00:19:22

2019-07-30 10:51:45

Markdown格式化文檔Linux

2023-08-25 14:47:56

TransFLACFLAC

2009-12-28 13:38:35

WPF類型轉換器

2018-03-23 09:29:56

深度學習神經網絡前端設計模型

2009-06-17 11:31:23

Open XMLUOF文檔

2021-05-05 10:06:09

React應用程序微前端

2009-09-11 12:41:41

C#類型轉換

2010-06-10 14:49:07

協議轉換器

2024-05-29 08:12:55

接口參數格式

2024-08-23 08:57:13

PyTorch視覺轉換器ViT

2009-07-15 16:56:59

Jython類型Java類型

2023-08-26 16:06:10

COBOLJava數據

2010-10-27 13:25:33

Oracle查詢
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 成人欧美一区二区三区黑人孕妇 | 国产欧美性成人精品午夜 | 精精久久| 国产精品久久久久久久久久久久 | 久久久久无码国产精品一区 | 亚洲午夜网 | 亚洲男女激情 | 国产成人精品一区二 | 久久ww| 免费在线黄色av | 中文字幕成人 | 欧洲精品一区 | 麻豆av免费观看 | 人人人人爽| 久久精品成人 | 九九精品热| 欧美久久一级特黄毛片 | 国产精品久久久久久久久久 | 亚洲一区二区三区在线 | www.9191| 久久国产欧美日韩精品 | 亚洲国产一区二区三区 | 亚洲一区二区三区四区五区午夜 | 九色在线观看 | 成年人的视频免费观看 | 欧美一区二区三区 | 亚洲综合天堂网 | 91婷婷韩国欧美一区二区 | 一区二区视频在线 | 久久爱综合 | 亚洲成人精品一区 | av色站| 国产在线观看免费 | 亚洲一区有码 | 国产农村一级片 | 一区二区中文字幕 | 拍真实国产伦偷精品 | 久久精品99 | 曰韩一二三区 | 久久精品av麻豆的观看方式 | 欧美日韩不卡合集视频 |