SpringMVC枚举类型字段处理

在日常的项目开发中经常会遇到一些取值范围固定的字段,例如性别、证件类型、会员等级等,此时我们可以利用枚举来最大程度减少字段的乱定义,统一管理枚举的值。

SpringMVC中对于枚举也有默认的处理策略:

  • 对于@RequestParam,Spring是通过ConverterFactory来处理的,大致处理策略是根据枚举名称或枚举下标来转换枚举。
    在这里插入图片描述

  • 对于@RequestBody,Spring是通过Jackson配置将json内的枚举值转换为对象的,大致处理策略同样是根据枚举名称或枚举下标来转换枚举。

在SpringMVC内对枚举的默认处理逻辑是根据枚举的类名或枚举下标来将请求参数转化为枚举对象,这显然不太灵活,因此我们需要调整枚举字段的处理逻辑。

RequestParam处理

我们可以自定义ConvertFactory来自定义枚举字段的转化策略。

  1. 定义BaseEnum接口,规定所有枚举都应该实现此接口

    public interface BaseEnum<T> {
    
        /**
         * 获取枚举值
         */
        T getCode();
    
        /**
         * 根据值获取对应的枚举
         * @param enumTypeClazz 枚举类型类
         * @param value 值
         */
        static <T extends BaseEnum> T getEnumByCode(Class<T> enumTypeClazz, Object value) {
            if (enumTypeClazz == null || value == null) {
                return null;
            }
    
            Optional<T> optional = Arrays.stream(enumTypeClazz.getEnumConstants()).filter(e ->{
                Object enumCode = e.getCode();
                return Objects.equals(Convert.convert(enumCode.getClass(), value),enumCode);
            }).findFirst();
    
            //如果不存在则抛异常
            return optional.orElseThrow( ()-> new RuntimeException("[" + enumTypeClazz.getSimpleName() + "]参数错误[" + value + "]"));
        }
    }
    
    
  2. 自定义ConverterFactory

    @Component
    public class EnumConverterFactory implements ConverterFactory<String, BaseEnum> {
        @Override
        public <T extends BaseEnum> Converter<String, T> getConverter(Class<T> targetType) {
            return source -> BaseEnum.getEnumByCode(targetType, source);
        }
    }
    
  3. 注册ConverterFactory

    @Configuration
    public class SpringMVCConfig implements WebMvcConfigurer {
    
        @Override
        public void addFormatters(FormatterRegistry registry) {
            registry.addConverterFactory(new EnumConverterFactory());
        }
    
    }
    
  4. 这样配置后请求参数就会自动转换为枚举了。

    //枚举类
    @AllArgsConstructor
    @Getter
    public enum Gender implements BaseEnum<Integer> {
        MALE(1,"男"),
        FEMALE(2,"女"),
            ;
        @EnumValue
        private Integer code;
        private String value;
    
    }
    
    //通过接口接受gender参数,能够根据code自动转换为对应的枚举
    @GetMapping("/test")
    public Gender insert(Gender gender) {
        return gender;
    }
    

RequestBody处理

RequestBody是通过Jackson转换对请求参数进行处理的,因此我们只需要自定义反序列化类即可

  1. 自定义序列化规则设置json内的值如何转换为枚举

    public class EnumDeserializer extends JsonDeserializer<BaseEnum> {
    
        /**
         * 根据参数值获取对应的枚举
         * @throws IOException
         * @throws JacksonException
         */
        @Override
        public BaseEnum deserialize(JsonParser p, DeserializationContext ctxt) throws IOException, JacksonException {
            // 当前值
            final String paramValue = p.getText();
    
            //获取序列化信息
            final JsonStreamContext parsingContext = p.getParsingContext();
            // 获取当前序列化的类的对象
            final Object currentValue = parsingContext.getCurrentValue();
            //获取当前序列化的字段名
            final String currentName = parsingContext.getCurrentName();
    
            try {
                // 反射获取当前序列化字段信息
                final Field declaredField = currentValue.getClass().getDeclaredField(currentName);
                // 通过字段信息获取对应的枚举的Class
                final Class<BaseEnum> targetType = (Class<BaseEnum>) declaredField.getType();
    
                //根据参数值获取对应的枚举
                BaseEnum baseEnum = BaseEnum.getEnumByCode(targetType, paramValue);
                if (ObjectUtil.isEmpty(baseEnum)) {
                    throw new RuntimeException("[" + currentName + "]参数错误");
                }
                //返回枚举
                return baseEnum;
            } catch (NoSuchFieldException e) {
                throw new RuntimeException("[" + currentName + "]参数错误");
            }
        }
    
    }
    
    1. 在枚举类加上 @JsonDeserialize(using = EnumDeserializer.class)
       //可以直接加到刚刚定义的BaseEnum接口上,这样所有枚举就自动继承了
       @JsonDeserialize(using = EnumDeserializer.class)
       public interface BaseEnum<T> {
       	……
       }
    
    1. 这样配置后@RequestBody就能够自动转换枚举了
    @PostMapping("/save")
    public User save(@RequestBody User user) {
       studentService.save(user);
       return user;
    }
    

枚举字段返回序列化

如果我们返回的对象内有枚举字段,SpringMVC会默认将枚举的名称作为值返回,如果我们想指定枚举类的某个属性作为值,可以通过@JsonValue指定

@AllArgsConstructor
@Getter
public enum Gender implements BaseEnum<Integer> {
    MALE(1,"男"),
    FEMALE(2,"女"),
        ;
	//指定转json时使用code作为值
    @JsonValue
    private Integer code;
    private String value;

}

或者直接在枚举类上加@JsonFormat,将枚举转换为对象格式

@JsonFormat(shape= JsonFormat.Shape.OBJECT)
@JsonDeserialize(using = EnumDeserializer.class)
public interface BaseEnum<T> {
	……
}

MybatisPlus对枚举的处理

MybatisPlus直接在枚举类的属性上加@EnumValue即可,并且兼容xml内的动态sql

@AllArgsConstructor
@Getter
public enum Gender implements BaseEnum<Integer> {
    MALE(1,"男"),
    FEMALE(2,"女"),
        ;
    //指定code作为入库时的值
    @EnumValue
    private Integer code;
    private String value;

}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/670160.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【用Python画画】六一儿童节画爱心

本文收录于 《Python编程入门》专栏&#xff0c;从零基础开始&#xff0c;分享一些Python编程基础知识&#xff0c;欢迎关注&#xff0c;谢谢&#xff01; 文章目录 一、前言二、代码示例三、知识点梳理四、总结 一、前言 本文介绍如何使用Python的海龟画图工具turtle&#xf…

Redis 线程模型

Redis 线程模型 背景简介Redis 单线程客户端发起 Redis 请求命令的工作原理单线程面临的挑战及问题 Redis 多线程Redis v4.0 多线程命令Redis v6.0 多线程网络模型 总结 背景 随着年龄的增长&#xff0c;很多曾经烂熟于心的技术原理已被岁月摩擦得愈发模糊起来&#xff0c;技术…

DNS设置(linux)

1.配置dns需要现在/etc/sysconfig/network-scripts/目录下的ifcfg-ens33(后面数字也可能是其他的)中配置DNS 2.编辑/etc/resolv.conf文件&#xff0c;将上面网卡中加的dns服务器ip添加到此文件 vi /etc/resolv.conf重启网络配置 service network restart常用的dns的ip 国内…

Flutter-自定义可展开文本控件

Flutter 在移动开发中&#xff0c;常常需要处理一些长文本显示的场景&#xff0c;如何优雅地展示这些文本并允许用户展开和收起是一个常见的需求。在本文中&#xff0c;我将分享如何使用Flutter实现一个可展开和收起的文本控件。 效果 我们将实现一个可展开和收起的文本控件…

JVM学习-详解类加载器(二)

双亲委派机制 双亲委派优势 避免类的重复加载&#xff0c;确保一个类的全局唯一性 Java类随着它的类加载器一起具备了一种带有优先级的层次关系&#xff0c;通过这种层次关系可以避免类的重复加载&#xff0c;当父类已经加载了该类&#xff0c;就没有必要子ClassLoader再加载…

Three.js——tween动画、光线投射拾取、加载.obj/.mtl外部文件、使用相机控制器

个人简介 &#x1f440;个人主页&#xff1a; 前端杂货铺 ⚡开源项目&#xff1a; rich-vue3 &#xff08;基于 Vue3 TS Pinia Element Plus Spring全家桶 MySQL&#xff09; &#x1f64b;‍♂️学习方向&#xff1a; 主攻前端方向&#xff0c;正逐渐往全干发展 &#x1…

低代码开发平台(Low-code Development Platform)的模块组成部分

低代码开发平台&#xff08;Low-code Development Platform&#xff09;的模块组成部分主要包括以下几个方面&#xff1a; 低代码开发平台的模块组成部分可以按照包含系统、模块、菜单组织操作行为等维度进行详细阐述。以下是从这些方面对平台模块组成部分的说明&#xff1a; …

计算机系统结构之FORK和JOIN

程序语言中用FORK语句派生并行任务&#xff0c;用JOIN语句对多个并发任务汇合。 FORK语句的形式为FORK m&#xff0c;其中m为新领程开始的标号。 JOIN语句的形式为JOIN n&#xff0c;其中n为并发进程的个数。 例1&#xff1a;给定算术表达式ZEA*B*C/DF经并行编译得到如下程序…

什么是 Redis 缓存?它解决了什么问题?怎么使用它?

前言 写在前面&#xff0c;让我们从 3 个问题开始今天的文章&#xff1a;什么是 Redis 缓存&#xff1f;它解决了什么问题&#xff1f;怎么使用它&#xff1f; 在笔者近 3 年的 Java 一线开发经历中&#xff0c;尤其是一些移动端、用户量大的互联网项目&#xff0c;经常会使用…

数学建模 —— 层次分析法(2)

目录 一、层次分析法&#xff08;AHP&#xff09; 二、构造比较判断矩阵 2.1 两两比较法 三、单准则下的排序及一致检验 3.1 单准则下的排序 3.2 一致性检验 四、层次总排序 4.1 层次总排序的步骤 4.2 总排序一致性检验 一、层次分析法&#xff08;AHP&#xff09; 方…

Centos 7部署NTP

介绍 NTP是Network Time Protocol&#xff08;网络时间协议&#xff09;的简称&#xff0c;它是用来通过互联网或局域网将计算机时钟同步到世界协调时间&#xff08;UTC&#xff09;的协议。 安装 # yum安装 yum install -y ntp# 离线安装 #下载地址&#xff1a;https://mir…

Caliburn.Micro框架学习笔记——多页面处理案例

在聊这个之前&#xff0c;我们先来看一个静态类 在 Caliburn.Micro 中&#xff0c;ViewLocator 是一个用于查找和关联视图与视图模型的静态类。默认情况下&#xff0c;它根据约定&#xff08;命名约定或其他规则&#xff09;自动找到与视图模型相对应的视图。然而&#xff0c;…

【5】MySQL数据库备份-XtraBackup - 全量备份

MySQL数据库备份-XtraBackup-全量备份 前言环境版本 安装部署下载RPM 包二进制包 安装卸载 场景分析全量备份 | 恢复备份恢复综合 增量备份 | 恢复部分备份 | 恢复 前言 关于数据库备份的一些常见术语、工具等&#xff0c;可见《MySQL数据库-备份》章节&#xff0c;当前不再重…

【普通切换】【DC-based handover】【DAPS】协议栈分析

移动网络切换 移动通信中切换是保证终端业务的基本流程&#xff0c;而切换时延是终端(UE)不能与任何基站交互(传递)用户面数据包的最短时间。 在5G(NR)网络中当终端(UE)接收到切换命令时&#xff0c;将断开与源小区的连接向目标小区发起随机接入过程。在此期间终端(UE)的数据传…

深入理解linux文件系统与日志分析12

一、inode和block 文件是存储在硬盘上&#xff0c;硬盘上的最小存储单位是扇区&#xff0c;每个扇区的大小是512字节。 inode&#xff1a;存储元信息&#xff08;文件的属性--权限、创建者、创建日期等等&#xff09; block&#xff1a;块 连续的八个扇区组成一个块&#xf…

rpmbuild多进程批量编译脚本

脚本用法 使用手册 bash spec.sh --help Please ensure the directory ~/rpmbuild exists, And the script can execute in any directory.Usage: bash spec.sh [OPTION]... [*.spec]... [OPTION]...or: bash spec.sh [*.spec]... [OPTION]... [*.spec]... rpmbuild all sp…

【原型模式】详解

一.概念 原型模式是一种创建型设计模式&#xff0c;它的主要思想是通过复制现有对象来创建新对象&#xff0c;而不是通过实例化一个类来创建。在原型模式中&#xff0c;我们称被复制的对象为原型&#xff08;Prototype&#xff09;&#xff0c;新创建的对象为克隆体&#xff0…

【OpenHarmony】TypeScript 语法 ③ ( 条件语句 | if else 语句 | switch case 语句 )

文章目录 一、条件语句1、if else 语句2、switch case 语句 参考文档 : <HarmonyOS第一课>ArkTS开发语言介绍 一、条件语句 1、if else 语句 TypeScript 中的 if 语句 / if else 语句 用法 , 与 JavaScript 语言中的 if 语句 / if else 语句 语法 基本相同 ; if else 语…

.gitignore 文件

一.什么是 .gitignore 文件 在任何当前工作的 Git 仓库中&#xff0c;每个文件都是这样的&#xff1a; 追踪的&#xff08;tracked&#xff09;- 这些是 Git 所知道的所有文件或目录。这些是新添加&#xff08;用 git add 添加&#xff09;和提交&#xff08;用 git commit 提…

Ubuntu 20.04安装CMake 3.22.6版本

Ubuntu 20.04通过apt安装的cmake版本是3.16.3&#xff0c;默认安装到/usr/bin/cmake路径。 $ cmake Command cmake not found, but can be installed with:sudo snap install cmake # version 3.29.3, or sudo apt install cmake # version 3.16.3-1ubuntu1.20.04.1See sna…