当前位置: 首页 > news >正文

SpringBoot中一行代码解决字符串向枚举类型转换的问题

1. 场景

在WEB开发,客户端和服务端传输的数据中经常包含一些这样的字段:字段的值只包括几个固定的字符串。 这样的字段意味着我们需要在数据传输对象(Data Transfer Object, DTO)中对该字段进行校验以避免客户端传输的非法数据持久化到我们的系统中。

public record UserCreateDto(String userName,// userType的值为NORMAL, SILVER_CARD, GOLD_CARD, String userType) {}

我们可以采用多种办法验证userType的正确性,如:

方法一:利用Validation和正则表达式进行验证

public record UserCreateDto(String userName,// userType的值为NORMAL, SILVER_CARD, GOLD_CARD@Pattern(regexp = "^NORMAL$|^SILVER_CARD$|^GOLD_CARD$")String userType) {
}

方法二:在代码中写validate方法,在使用到DTO代码中调用validate方法

public record UserCreateDto(String userName,// userType的值为NORMAL, SILVER_CARD, GOLD_CARDString userType) {public void validate() {if (List.of("NORMAL", "SILVER_CARD", "GOLD_CARD").contains(userType)) {return;}throw new IllegalArgumentException("userType must be NORMAL, SILVER_CARD, GOLD_CARD");}
}

比较这两种方法,两种方法各有优缺点:

优点缺点
方法一在DTO创建时(即参数的入口处)就可以验证数据的有效性在@Pattern中使用字符串常量不方便,意味着开发者很难在整个代码中使用统一的自定义常量,为后期的修改带来不便
方法二开发者可以在整个代码中使用统一的自定义常量,方便后续的修改需要开发者主动调用validate方法,容易遗漏调用

2. 面向对象的解决办法

可能你早已想到用枚举来解决上述场景中的问题,没错,在面向对象编程中,枚举是解决这种问题的最好的解决办法。

public enum UserType {NORMAL, SILVER_CARD, GOLD_CARD
}public record UserCreateDto(String userName,// userType的值为NORMAL, SILVER_CARD, GOLD_CARD@NotNullUserType userType) {
}

枚举让我们的参数具有类型约束,并且具有可复用性易修改等特性。

但是在SpringBoot中默认是不支持String到Enum的转换(读者可以尝试一下,不管客户端传入的userType正确与否,在DTO中userType值均为null )。

为了解决这个问题很多开发者都是通过自定义Conveter来进行String到Enum的转换的。如此常见的场景,作为开发者的我们都能想到使用统一的Converter,难到作为框架的开发者想不到?

3. 一行代码解决String到Enum的转换问题

先上解决方案。

@Configuration
public class WebMvcConfiguration implements WebMvcConfigurer {@Overridepublic void addFormatters(FormatterRegistry registry) {// 通过ApplicationConversionService向应用中注入ConverterApplicationConversionService.configure(registry);}

就是这么简单,在应用中自定义WebMvcConfigurer,覆写addFormatters方法,并通过ApplicationConversionService向应用中注入String到Enum的Converter。

4. 原理分析

通过分析ApplicationConversionService的时序图,我们可以看到ApplicationConversionService最终通过DefaultConversonService调用ConverterRegister向应用注册了StringToEnumConverterFactory,从名字可以看出来StringToEnumConverterFactory就是负责String向Enum转换的。
在这里插入图片描述
StringToEnumConverterFactory的代码如下:

final class StringToEnumConverterFactory implements ConverterFactory<String, Enum> {@Overridepublic <T extends Enum> Converter<String, T> getConverter(Class<T> targetType) {return new StringToEnum(ConversionUtils.getEnumType(targetType));}private static class StringToEnum<T extends Enum> implements Converter<String, T> {private final Class<T> enumType;StringToEnum(Class<T> enumType) {this.enumType = enumType;}@Override@Nullablepublic T convert(String source) {if (source.isEmpty()) {// It's an empty enum identifier: reset the enum value to null.return null;}return (T) Enum.valueOf(this.enumType, source.trim());}}
}

可以看出,StringToEnumConverterFactory中也是通过Enum的valueOf方法完成String到Enum的转换的。

5. 方案的不足

采用Spring框架提供的StringToEnum Converter带给我们便利性的同时,也存在一些约束,如:

  • Enum中实例的大小写必须和字符串的大小写一致,如字符串是小写的normal、silver_card、gold_card,Enum定义的实例也必须是normal、silver_card、gold_card,这个可能并不符合代码规范(通常Enum的实例都要球全大写);
  • 字符串中包含一些特殊字符是Java命名规范不允许的,如中划线。

因此,选用哪种方法完成字符串到Enum的转换还要根据实际的应用场景出发。

http://www.lryc.cn/news/32171.html

相关文章:

  • Praat之基频分析
  • 乡村企业门户网站
  • Deploy Workshop|DIY部署环境,让OceanBase跑起来
  • 【CPP】定义一个类
  • 谷歌广告投放步骤流程是什么?一文带你全方位了解实操细节
  • TypeScript 怎么去查找类型定义的?
  • NPM包管理器
  • IT英语记录
  • SRS4.0 源码分析- RTC模块相关类
  • 数位DP
  • 剑指offer(一)-链表
  • CDH大数据平台入门篇之搭建与部署
  • Spark Join
  • 数字的转化规则?
  • MySQL面试题-锁相关
  • Windows 终端编译 C代码
  • SpringCloud:Feign的使用及配置
  • Parquet学习与使用之BloomFilter的应用
  • 95%置信区间计算-理解
  • 深度学习pytorch实战三:VGG16图像分类篇自建数据集图像分类三类
  • 2023年3月软考高项(信息系统项目管理师)报名走起!!!
  • 模电学习11 运算放大器学习入门
  • spring学习3.5
  • 名创优品:国内“触礁”,海外“提速”
  • Java学习笔记 --- Tomcat
  • 面向对象设计模式:行为型模式之状态模式
  • 【Python入门第二十五天】Python 作用域
  • 运行时数据区及程序计数器
  • 手写操作系统+文件系统开源啦
  • 小众但意外觉得蛮好用的剪辑软件!纯良心分享