年前, Mybatis
官方已经发布了 3.5.0
,从 Mybatis-Plus
的 commit
记录可看出,也在做了一些相关的适配工作,但迟迟未发布 3.0.8
的正式版,所以到现在只有静静的等待官方发布啦,不过我们可以通过源码打包自己优先体验.
废话不多说,直接进入正题,原来处理枚举需要必须通过 Mybatis-Plus
的扩展配置 typeEnumsPackage
来进行处理,但 3.0.8
发布之后,直接可以使用 Mybatis
的配置方式了,也就是设置 defaultEnumTypeHandler
.
注意下, EnumAnnotationTypeHandler
已经过时了,所以在 3.0.8
以后,只需要使 EnumTypeHandler
即可处理 Mybatis-Plus
提供的两种枚举处理方案(实现 IEnum
接口或使用 @EnumValue
注解).
mybatis-plus: configuration: default-enum-type-handler: com.baomidou.mybatisplus.extension.handlers.EnumTypeHandler复制代码
<bean id="sqlSessionFactory" class="com.baomidou.mybatisplus.extension.spring.MybatisSqlSessionFactoryBean"> <property name="configuration"> <bean class="com.baomidou.mybatisplus.core.MybatisConfiguration"> <property name="defaultEnumTypeHandler" value="com.baomidou.mybatisplus.extension.handlers.EnumTypeHandler"/> </bean> </property> </bean>复制代码
到这里也许你会产生疑问了,使用这种方案和配置 typeEnumsPackage
的区别是什么?
//com.baomidou.mybatisplus.extension.spring.MybatisSqlSessionFactoryBean 525行 TypeHandlerRegistry typeHandlerRegistry = configuration.getTypeHandlerRegistry(); classes.stream().filter(Class::isEnum) //是枚举 //是IEnum实现类或使用@EnumValue注解 .filter(cls -> IEnum.class.isAssignableFrom(cls) || EnumTypeHandler.dealEnumType(cls).isPresent()) //注册枚举转换器,注意下register方法,这里会反射调用EnumTypeHandler的默认构造方法 .forEach(cls -> typeHandlerRegistry.register(cls, EnumTypeHandler.class));复制代码
看上面这代码是片段是配置 typeEnumsPackage
时进行的处理,然后再结合下面 EnumTypeHandler
的构造方法.
//com.baomidou.mybatisplus.extension.handlers.EnumTypeHandler public EnumTypeHandler(Class<E> type) { if (type == null) { throw new IllegalArgumentException("Type argument cannot be null"); } this.type = type; //是IEnum实现类 if (IEnum.class.isAssignableFrom(type)) { try { //使用getValue方法获取值 this.method = type.getMethod("getValue"); } catch (NoSuchMethodException e) { throw new IllegalArgumentException(String.format("NoSuchMethod getValue() in Class: %s.", type.getName())); } } else { // 这里会缓存了枚举类对应的方法 this.method = TABLE_METHOD_OF_ENUM_TYPES.computeIfAbsent(type, k -> { // 寻找标记@EnumValue字段 Field field = dealEnumType(this.type).orElseThrow(() -> new IllegalArgumentException(String.format("Could not find @EnumValue in Class: %s.", type.getName()))); // 获取标记@EnumValue字段的get方法 return ReflectionKit.getMethod(this.type, field); }); } }复制代码
如果我们配置 typeEnumsPackage
,对使用注解的方式来说的话,就等同于预先初始化了缓存,但对于使用实现 IEnum
接口的方式来说,并无多大作用.
所以,如果你是使用注解的方式且不想懒加载,你可以配置 typeEnumsPackage
来初始化枚举缓存,其他的方式的话就直接设置 defaultEnumTypeHandler
即可.