MybatisPlus中Enum的使用(MybatisEnumTypeHandler)及遇到的问题

1.基本使用

简单的使用可以参照官网的文档,三步走

第一步,要用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

2.我常用的方式

上面的方法也还有别的实现方式,再不改变开发依赖的情况下,能变动的就是扫描方式呗,对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>

3.使用中遇到的一些问题

请注意,当项目中同时使用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 <T> TypeHandler<T> 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<E> 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<String, String> 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<String, String> 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,可以设置对应参数解决

我得经验也不是很多,对于源码理解不深刻,如有错误还请各位大佬指出;

上述问题解决方案也并不完美,只是希望能给大家提供一个解决思路。

  • 1
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
很高兴回答您关于Javaenum类型的问题。在Javaenum类型是一种特殊的数据类型,用于定义一组命名常量。下面是一个示例代码及详解: ```java public class EnumExample { // 定义一个enum类型 public enum Day { MONDAY("星期一"), TUESDAY("星期二"), WEDNESDAY("星期三"), THURSDAY("星期四"), FRIDAY("星期五"), SATURDAY("星期六"), SUNDAY("星期日"); private String chineseName; // 枚举常量的文名称 // 构造方法 Day(String chineseName) { this.chineseName = chineseName; } // 获取枚举常量的文名称 public String getChineseName() { return chineseName; } } public static void main(String[] args) { // 遍历枚举常量并输出文名称 for (Day day : Day.values()) { System.out.println(day + ":" + day.getChineseName()); } } } ``` 在上述代码,我们定义了一个名为`Day`的enum类型。该enum类型包含了一组枚举常量,分别代表一周的不同天。每个枚举常量都有一个关联的文名称,通过构造方法传递并保存在`chineseName`字段。 在`main`方法,我们使用`Day.values()`方法获取所有枚举常量,并通过增强型`for`循环遍历每个枚举常量。在循环,我们打印出每个枚举常量的名称和对应的文名称。 运行以上代码,您将会看到以下输出: ``` MONDAY:星期一 TUESDAY:星期二 WEDNESDAY:星期三 THURSDAY:星期四 FRIDAY:星期五 SATURDAY:星期六 SUNDAY:星期日 ``` 通过使用enum类型,我们可以定义一组具有固定值的常量,并且可以为每个常量添加其他属性和方法。这使得代码更加清晰、可读性更高,并且在编译期间会进行类型检查,避免了一些错误。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值