自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

SpringBoot 使用轉(zhuǎn)換器將前端參數(shù)轉(zhuǎn)換為枚舉

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

[[421671]]

前言

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

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

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

1實現(xiàn)過程

配置轉(zhuǎn)換器

  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 里實現(xiàn) addFormatters 方法即可,然后 new 一個 ConverterFactory。

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

就這些,很簡單的實現(xiàn)。下面介紹下項目的內(nèi)容和代碼,方便理解。

項目代碼

  • 請求參數(shù):
  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("請求參數(shù):{}", 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("返回參數(shù):{}", 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 類型的枚舉。

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

為什么要這么用呢?

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

  • 枚舉
  1. public interface BaseEnum { 
  2.     Object getCode(); 
  1. public enum OrderStatusEnum implements BaseEnum { 
  2.  
  3.     INIT(0, "初始狀態(tài)"), 
  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,所有的枚舉都繼承這個接口,并實現(xiàn) getCode 方法。

@EnumValue:MyBatis-Plus 的枚舉,和數(shù)據(jù)庫字段映射用的

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

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

效果

測試結(jié)果

測試結(jié)果經(jīng)過驗證,是可以勝任傳入數(shù)值和字符串的。

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

具體說明

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

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

在 ConverterFactory 的 getConverter 方法則需要返回一個實際的轉(zhuǎn)換器 Converter

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

convert 方法的入?yún)⑹且粋€ source,就是要轉(zhuǎn)換為什么類型的,這里就是數(shù)字/字符串,然后返回一個枚舉即可。

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

2優(yōu)化

一般 WebConfig 中除了實現(xiàn) addFormatters 方法外,還會實現(xiàn) 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.         // 枚舉轉(zhuǎn)換 
  31.         registry.addConverterFactory(enumConverterFactory); 
  32.     } 

這種就需要咱們創(chuàng)建 EnumConverterFactory 類并實現(xiàn) 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總結(jié)

當(dāng)然這里也有一些其他的優(yōu)化點,比如可以使用緩存將 Convert 緩存起來。

 

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

 

責(zé)任編輯:武曉燕 來源: 程序員小航
相關(guān)推薦

2010-06-10 14:33:03

協(xié)議轉(zhuǎn)換器

2013-06-13 15:10:27

.NET代碼轉(zhuǎn)換

2010-06-10 14:44:33

協(xié)議轉(zhuǎn)換器

2010-06-10 14:38:30

協(xié)議轉(zhuǎn)換器

2010-06-10 15:03:13

協(xié)議轉(zhuǎn)換器

2014-05-04 12:51:21

Javascript編譯器

2019-07-30 10:51:45

Markdown格式化文檔Linux

2023-05-05 00:19:22

2023-08-25 14:47:56

TransFLACFLAC

2018-03-23 09:29:56

深度學(xué)習(xí)神經(jīng)網(wǎng)絡(luò)前端設(shè)計模型

2009-12-28 13:38:35

WPF類型轉(zhuǎn)換器

2009-06-17 11:31:23

Open XMLUOF文檔

2021-05-05 10:06:09

React應(yīng)用程序微前端

2009-09-11 12:41:41

C#類型轉(zhuǎn)換

2010-06-10 14:49:07

協(xié)議轉(zhuǎn)換器

2024-05-29 08:12:55

接口參數(shù)格式

2023-08-26 16:06:10

COBOLJava數(shù)據(jù)

2009-07-15 16:56:59

Jython類型Java類型

2010-10-27 13:25:33

Oracle查詢

2024-08-23 08:57:13

PyTorch視覺轉(zhuǎn)換器ViT
點贊
收藏

51CTO技術(shù)棧公眾號