MyBatis-Plus通用枚举

MyBatis-Plus通用枚举

MyBatis-Plus通用枚举是什么

MyBatis-Plus 通用枚举是基于 MyBatis 的类型处理器 TypeHandler 扩展实现的标准化、通用化枚举适配方案,核心定位是解决代码层枚举语义化与数据库层编码轻量化的映射问题。

它通过注解(@EnumValue)或接口(IEnum)约定枚举值与数据库存储编码的映射规则,结合枚举包扫描机制,实现枚举类型与数据库字段编码的无侵入式自动化双向转换,无需开发者编写手动转换逻辑,且一套规则可统一适配所有符合约定的枚举类。

示例详解

下面通过示例对MyBatis-Plus通用枚举的执行流程进行详细介绍

数据库表添加字段sex

创建通用枚举类型

用 @EnumValue 标注数据库存储的枚举编码字段

java 复制代码
@Getter
public enum SexEnum {
    MALE(1, "男"),
    FEMALE(2, "女");

    @EnumValue
    private Integer sex;      // 核心字段
    private String sexName;   // 业务描述字段

    SexEnum(Integer sex, String sexName) {
        this.sex = sex;
        this.sexName = sexName;
    }
}

MyBatis-Plus 通用枚举类的核心构成要素如下:

首先需通过 @Getter 注解(或手动编写)为所有私有字段提供 Getter 方法,满足 MP 反射获取字段值的基础要求;其次必须定义由 @EnumValue 注解标注的数据库映射核心字段,该字段是 MP 实现枚举和数据库编码双向转换的唯一依据,写库时反射获取其编码值写入数据库,读库时依据编码值反向匹配枚举项;可写业务描述字段(非 MP 强制要求,可提升枚举的业务语义可读性);此外需声明具体枚举项,每个枚举项对应一组数据库编码值和业务描述值,枚举项将抽象的业务状态转化为代码中语义明确、可直接引用的实例;最后通过私有构造器完成核心字段与业务描述字段的初始化。

在 MyBatis-Plus 早期版本中,实现通用枚举的主流方式是让枚举类实现 IEnum 接口,且需指定接口泛型(泛型类型与数据库编码字段类型一致),并强制重写 getValue() 方法,下面是具体代码

java 复制代码
@Getter
public enum SexEnum1 implements IEnum<Integer> {
    MALE(1, "男"),
    FEMALE(2, "女");

    private Integer sex;
    private String sexName;

    SexEnum1(Integer sex, String sexName) {
        this.sex = sex;
        this.sexName = sexName;
    }

    @Override
    public Integer getValue() {
        return this.sex;
    }
}

新版仍兼容该方式,若枚举同时标注 @EnumValue 和实现 IEnum 接口,@EnumValue 的优先级更高

实体类

java 复制代码
@Data
@NoArgsConstructor
@AllArgsConstructor
public class User {

    private Long id;
    private String name;
    private Integer age;
    private String email;
    // 枚举
    private SexEnum sex;

}

实体类中的 sex 属性无需定义为数据库存储的编码类型,可直接声明为自定义的通用枚举类型,MyBatis-Plus 框架会自动完成枚举类型与数据库编码之间的双向转换

type-enums-package 是 MyBatis-Plus 配置文件中用于指定通用枚举类扫描包路径的配置项。MP 启动阶段会扫描该路径下所有符合通用枚举规范的枚举类,一方面自动识别枚举类中基于@EnumValue / IEnum 定义的映射规则,另一方面为这些枚举类自动绑定内置的 MybatisEnumTypeHandler 类型处理器,而 MybatisEnumTypeHandler 作为核心组件,支撑了通用枚举与数据库编码之间的自动双向转换能力。

测试代码

java 复制代码
    @Autowired
    private UserMapper userMapper;

    @Test
    public void testSexEnum(){
        User user = new User();
        user.setName("Enum");
        user.setAge(20);
        user.setEmail("test@enum.com");
        user.setSex(SexEnum.MALE);
        userMapper.insert(user);
        System.out.println(user.getSex());    // 获取枚举实例
        System.out.println(user.getSex().getSex()); // 获取数据库存储的编码值
        System.out.println(user.getSex().getSexName());  // 获取业务描述
    }

执行 user.setSex(SexEnum.MALE); MP 会通过 MybatisEnumTypeHandler 自动将枚举项 SexEnum.MALE 转换为其 @EnumValue 标注的编码值 1 写入数据库。

user.getSex() 获取 SexEnum.MALE 枚举实例

user.getSex().getSex() 获取数据库存储的编码值

user.getSex().getSexName() 获取业务描述

执行结果如图:


相关推荐
abc123456sdggfd11 小时前
HTML5中Vuex持久化插件中WebStorage的底层配置
jvm·数据库·python
pele11 小时前
Go语言如何发GET请求_Go语言HTTP GET请求教程【总结】
jvm·数据库·python
云烟成雨TD11 小时前
Spring AI Alibaba 1.x 系列【33】Human-in-the-Loop(人在回路)演示
java·人工智能·spring
weixin_5806140011 小时前
Go 语言中 go install 命令的正确用法与常见误区详解
jvm·数据库·python
qq_6543669811 小时前
Bootstrap 5移除jQuery依赖 Bootstrap 5如何不使用jQuery
jvm·数据库·python
今天你TLE了吗11 小时前
LLM到Agent&RAG——AI概念概述 第五章:Skill
人工智能·笔记·后端·学习
m0_6765443812 小时前
CSS如何实现元素悬浮在页面底部_利用fixed定位与底部间距
jvm·数据库·python
weixin_5689960612 小时前
Redis怎样监控当前发生了多少次内存驱逐
jvm·数据库·python
2301_7965885012 小时前
CSS如何制作导航栏平滑移动_使用transition与left属性
jvm·数据库·python
難釋懷12 小时前
Redis服务器端优化-内存划分和内存配置
java·redis·spring