年前,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
注解).
spring-boot
mybatis-plus:
configuration:
default-enum-type-handler: com.baomidou.mybatisplus.extension.handlers.EnumTypeHandler复制代码
springmvc
<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
即可.