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() 获取业务描述

执行结果如图:


相关推荐
暹罗软件开发2 小时前
多线程协作利器:CountDownLatch 核心用法与场景解析
后端
天天摸鱼的java工程师2 小时前
支付回调处理,咱得整得 “幂等可靠” 不翻车
java·后端
踏浪无痕2 小时前
高并发写入 API 设计:借鉴 NSQ 的内存队列与背压机制
后端·面试·go
⑩-2 小时前
Spring 事务失效
java·后端·spring
编织幻境的妖2 小时前
数据库物化视图与普通视图区别
数据库·oracle
BingoGo2 小时前
告别 Shell 脚本:用 Laravel Envoy 实现干净可复用的部署
后端
爱因斯坦乐2 小时前
【若依】前后端分离添加导入
java·前端·javascript
Cache技术分享2 小时前
267. Java 集合 - Java 开发必看:ArrayList 与 LinkedList 的全方位对比及选择建议
前端·后端
用户8307196840822 小时前
Spring Boot JWT登录授权使用指南(无感刷新)
java·spring boot