简单的使用可以参照官网的文档,三步走
第一步,要用Enum就要先创建一个,我这里随便写了个,要注意的是两个注解
@EnumValue 这个是mybatisplus的注解,代表如果使用Enum作为实体类中字段的类型,那会找到对应Enum中标识为@EnumValue的字段存入数据库
@JsonValue 这个是jackson的注解,是把此注解标记的值返回给前端。如果用gson或者fastjson也会有对应的方式,此处不赘述
这里也可以实现IEnum接口,效果是一样的,看具体情况
- import com.baomidou.mybatisplus.annotation.EnumValue;
- import com.fasterxml.jackson.annotation.JsonValue;
- import lombok.Getter;
- import lombok.RequiredArgsConstructor;
-
- @Getter
- @RequiredArgsConstructor
- public enum TemplateEnum {
- /**
- * 通用模板
- */
- GENERIC("通用模板"),
- /**
- * 专用模板
- */
- DEDICATED("专用模板");
-
- @EnumValue
- @JsonValue
- private final String desc;
- }
第二步,实体类要用使用对应的枚举类
可以看到templateType字段是TemplateEnum类型的
- import com.baomidou.mybatisplus.annotation.TableField;
- import com.baomidou.mybatisplus.annotation.TableId;
- import com.baomidou.mybatisplus.annotation.TableName;
- import com.baomidou.mybatisplus.core.handlers.MybatisEnumTypeHandler;
- import io.swagger.annotations.ApiModel;
- import io.swagger.annotations.ApiModelProperty;
- import lombok.Data;
- import org.apache.ibatis.type.JdbcType;
-
- import java.io.Serializable;
- import java.time.LocalDateTime;
-
-
- @TableName(value = "template")
- @Data
- @ApiModel(value = "Template对象", description = "模板表")
- public class Template implements Serializable {
-
- private static final long serialVersionUID = 1L;
-
- @ApiModelProperty("模板编码")
- @TableId
- private String templateCode;
-
- @ApiModelProperty("模板类型")
- private TemplateEnum templateType;
-
- }
第三步
配置扫描路径,就是enum所在的包,也可以具体到某个类。用;分割
目前我使用的是mybatisplus3.5.1,默认的typeHandler是MybatisEnumTypeHandler,所以这里也可以不用设置
- mybatis-plus:
- # 支持统配符 * 或者 ; 分割
- typeEnumsPackage: com.baomidou.springboot.entity.enums
上面的方法也还有别的实现方式,再不改变开发依赖的情况下,能变动的就是扫描方式呗,对mybatisplus来说配置可以做的事情注入式也可以实现。
上面的第三步不用了 ,然后又分两种情况,BaseMapper方式和Mapper.xml方式,其实主要是看你的sql语句在哪
BaseMapper就是说使用mybatisplus带的IService或者BaseMapper实现好的方法
我们都知道如果要使用selectById方法,要在实体类使用@TableId注解才行,可以说实体类和BaseMapper是绑定的。
- @TableName(value = "template",autoResultMap = true)
- @Data
- @ApiModel(value = "Template对象", description = "模板表")
- public class Template implements Serializable {
-
- private static final long serialVersionUID = 1L;
-
- @ApiModelProperty("模板编码")
- @TableId
- private String templateCode;
-
- @ApiModelProperty("模板类型")
- @TableField(typeHandler = MybatisEnumTypeHandler.class)
- private TemplateEnum templateType;
-
- }
另外一种就是在xml中加入对应的typeHandler
- <resultMap id="base" type="com.xxx.Template">
- <result column="template_code" property="templateCode"/>
- <result column="template_type" property="templateType" typeHandler="com.baomidou.mybatisplus.core.handlers.MybatisEnumTypeHandler"/>
- resultMap>
请注意,当项目中同时使用BaseMapper方式和Mapper.xml方式并且都有注入对应typeHandler的时候是可以的,但是不能再使用配置文件扫描整个包,这样会和字段上定义的typeHandler产生冲突报错。
按照上述方式2进行配置,可能会出现插入时javaType不匹配报错
在进行bean注入的时候,我们要创建MybatisEnumTypeHandler,需要用TypeHandlerRegistry类中的getInstance反射进行创建,这里有个重要的参数就是javaType,这个参数可以在实体类或xml中进行配置。
下面是TypeHandlerRegistry部分代码,其中javaTypeClass会根据是否配置javaType进行变化,如果javaType = true,那会按照实体类或者xml对应字段的java类型获取,如果javaType = false,那这里有可能是Object或者null,对于MybatisEnumTypeHandler来说应该是Object(我调试的时候是这样)
- public
TypeHandler getInstance(Class> javaTypeClass, Class> typeHandlerClass) { - Constructor c;
- if (javaTypeClass != null) {
- try {
- c = typeHandlerClass.getConstructor(Class.class);
- return (TypeHandler)c.newInstance(javaTypeClass);
- } catch (NoSuchMethodException var5) {
- } catch (Exception var6) {
- throw new TypeException("Failed invoking constructor for handler " + typeHandlerClass, var6);
- }
- }
-
- try {
- c = typeHandlerClass.getConstructor();
- return (TypeHandler)c.newInstance();
- } catch (Exception var4) {
- throw new TypeException("Unable to find a usable constructor for " + typeHandlerClass, var4);
- }
- }
如果注册是Object,那么对于MybatisEnumTypeHandler的构造函数来说
- public MybatisEnumTypeHandler(Class
enumClassType) { - if (enumClassType == null) {
- throw new IllegalArgumentException("Type argument cannot be null");
- } else {
- this.enumClassType = enumClassType;
- MetaClass metaClass = MetaClass.forClass(enumClassType, REFLECTOR_FACTORY);
- String name = "value";
- if (!IEnum.class.isAssignableFrom(enumClassType)) {
- name = (String)findEnumValueFieldName(this.enumClassType).orElseThrow(() -> {
- return new IllegalArgumentException(String.format("Could not find @EnumValue in Class: %s.", this.enumClassType.getName()));
- });
- }
-
- this.propertyType = ReflectionKit.resolvePrimitiveIfNecessary(metaClass.getGetterType(name));
- this.getInvoker = metaClass.getGetInvoker(name);
- }
- }
propertyType接收到的就是Object,在SqlSourceBuilder创建参数映射的时候可能就会找不到对的参数类型
- private ParameterMapping buildParameterMapping(String content) {
- Map
propertiesMap = this.parseParameterMapping(content); - String property = (String)propertiesMap.get("property");
- Class propertyType;
- if (this.metaParameters.hasGetter(property)) {
- propertyType = this.metaParameters.getGetterType(property);
- } else if (this.typeHandlerRegistry.hasTypeHandler(this.parameterType)) {
- propertyType = this.parameterType;
- } else if (JdbcType.CURSOR.name().equals(propertiesMap.get("jdbcType"))) {
- propertyType = ResultSet.class;
- } else if (property != null && !Map.class.isAssignableFrom(this.parameterType)) {
- MetaClass metaClass = MetaClass.forClass(this.parameterType, this.configuration.getReflectorFactory());
- if (metaClass.hasGetter(property)) {
- propertyType = metaClass.getGetterType(property);
- } else {
- propertyType = Object.class;
- }
- } else {
- propertyType = Object.class;
- }
-
- Builder builder = new Builder(this.configuration, property, propertyType);
- Class> javaType = propertyType;
- String typeHandlerAlias = null;
- Iterator var8 = propertiesMap.entrySet().iterator();
-
- while(var8.hasNext()) {
- Entry
entry = (Entry)var8.next(); - String name = (String)entry.getKey();
- String value = (String)entry.getValue();
- if ("javaType".equals(name)) {
- javaType = this.resolveClass(value);
- builder.javaType(javaType);
- } else if ("jdbcType".equals(name)) {
- builder.jdbcType(this.resolveJdbcType(value));
- } else if ("mode".equals(name)) {
- builder.mode(this.resolveParameterMode(value));
- } else if ("numericScale".equals(name)) {
- builder.numericScale(Integer.valueOf(value));
- } else if ("resultMap".equals(name)) {
- builder.resultMapId(value);
- } else if ("typeHandler".equals(name)) {
- typeHandlerAlias = value;
- } else if ("jdbcTypeName".equals(name)) {
- builder.jdbcTypeName(value);
- } else if (!"property".equals(name)) {
- if ("expression".equals(name)) {
- throw new BuilderException("Expression based parameters are not supported yet");
- }
-
- throw new BuilderException("An invalid property '" + name + "' was found in mapping #{" + content + "}. Valid properties are " + "javaType,jdbcType,mode,numericScale,resultMap,typeHandler,jdbcTypeName");
- }
- }
-
- if (typeHandlerAlias != null) {
- builder.typeHandler(this.resolveTypeHandler(javaType, typeHandlerAlias));
- }
-
- return builder.build();
- }
所以如果使用MybatisEnumTypeHandler报错有关javatype,可以设置对应参数解决
我得经验也不是很多,对于源码理解不深刻,如有错误还请各位大佬指出;
上述问题解决方案也并不完美,只是希望能给大家提供一个解决思路。