Mybatis-TypeHandler类型处理器

TypeHandler

处理数据库类型和Java类型之间的转换。

在远古时期我们学习过jdbc, 那时候读取数据库需要手写从ResultSet通过columnName或者columnIndex读取数据。现在用上框架了也不要忘记老本行,mybatis底层同样是这么干的,我们看一下源码。抽象类BaseTypeHandler实现的TypeHandler的getResult方法。

@Override
  public T getResult(ResultSet rs, String columnName) throws SQLException {
    try {
      return getNullableResult(rs, columnName);
    } catch (Exception e) {
      throw new ResultMapException("Error attempting to get column '" + columnName + "' from result set.  Cause: " + e, e);
    }
  }

其中getNullableResult(rs, columnName);就是读取数据,是不是很熟悉,点进去看一下他是怎么写的。

public abstract T getNullableResult(ResultSet rs, String columnName) throws SQLException;

这是个抽象方法,这个BaseTypeHandler就是给具体的类型转换器集成的,这样子类只需要实现getNullableResult方法就能只写getResult具体处理过程,不需要处理异常。类似于模板方法设计模式,这是种将具体操作延迟到子类的做法。我们随意挑选一个Mybatis的类型转换器BigIntegerTypeHandler看一下他是怎么实现的(其他代码没粘贴出来,点代码打开点进去看看 非常简单)。

@Override
  public BigInteger getNullableResult(ResultSet rs, String columnName) throws SQLException {
    BigDecimal bigDecimal = rs.getBigDecimal(columnName);
    return bigDecimal == null ? null : bigDecimal.toBigInteger();
  }

直接使用了ResultSet::getBigDecimal,这个例子虽然不太直观但简单粗暴的看到了类型转换器的具体实现。当你觉得mybatis提供的这些转换器不够用时,我们可以自己自定义一个类型转换器,最终在你的mapper.xml中指定,就可以完成类型转换了。实操一下。
简单的假设你现在实体类中有一个枚举类型想要自动转换,我们自定义一个枚举转换器。同样是一个不恰当的例子但能演示功能。
枚举类GenderEnum

@Getter
@AllArgsConstructor
@ToString
public enum GenderEnum {

    MALE("1", "男"),
    FEMALE("2", "女"),
    ;

    private final String code;
    private final String desc;
}

实体类User.java

@TableName("user")
@Slf4j
@Getter
@Setter
public class User implements Serializable {
    @TableId
    String id;
    @TableField("username")
    String username;
    @TableField("pswd")
    String pswd;
    @TableField("gender")
    GenderEnum gender;
}

自定义类型转换器GenderEnumTypeHandler

public class GenderEnumTypeHandler extends BaseTypeHandler<GenderEnum> {

    @Override
    public void setNonNullParameter(PreparedStatement preparedStatement, int i, GenderEnum genderEnum, JdbcType jdbcType) throws SQLException {
        preparedStatement.setString(i, genderEnum.getCode());
    }

    @Override
    public GenderEnum getNullableResult(ResultSet resultSet, String columName) throws SQLException {
        String code = resultSet.getString(columName);
        return GenderEnum.parseGenderByCode(code);
    }

    @Override
    public GenderEnum getNullableResult(ResultSet resultSet, int index) throws SQLException {
        String code = resultSet.getString(index);
        return GenderEnum.parseGenderByCode(code);
    }

    @Override
    public GenderEnum getNullableResult(CallableStatement callableStatement, int index) throws SQLException {
        String code = callableStatement.getString(index);
        return GenderEnum.parseGenderByCode(code);
    }
}

UserMapper.xml,在resultMap中对性别枚举指定typeHandler为自己写的类型转换器。
mxl中指定typeHandler
运行看看类型转换成功了没
在这里插入图片描述
好了,关于typeHandler的分享就到这里了,代码写的很简陋,实际会封装的复杂一点。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
mybatis-puls报错java.lang.illegalstateexception: no typehandler found fo的原因是没有找到合适的类型处理器。 MyBatis-Puls是MyBatis的一个插件,用于增强MyBatis框架的功能。在使用MyBatis-Puls的过程中,如果出现了"no typehandler found fo"的错误,通常是由于没有正确配置类型处理器所致。 类型处理器(TypeHandler)是MyBatis中用于将数据库中的数据类型与Java中的数据类型进行转换的一个重要组件。当MyBatis-Puls在映射对象和数据库表之间进行数据转换时,会根据声明的类型处理器来进行相应的转换操作。 要解决这个错误,我们需要确保在MyBatis-Puls的配置文件中正确配置了类型处理器。 首先,确认错误信息中提到的类型处理器确实存在于项目中。如果没有找到合适的类型处理器,我们可以自定义一个类型处理器,实现TypeHandler接口,并在配置文件中进行注册。 其次,确认类型处理器的配置是否正确。在MyBatis-Puls的配置文件中,可以通过typeHandlers标签来配置类型处理器。确保需要的类型处理器已经被正确配置。 最后,检查映射对象的属性与数据库字段的类型是否匹配。如果属性的类型与数据库字段的类型不一致,也会导致类型处理器报错。确保属性和字段的类型匹配,或者在MyBatis-Puls的配置文件中配置适当的类型处理器来解决类型转换问题。 总之,当出现mybatis-puls报错java.lang.illegalstateexception: no typehandler found fo时,需要检查是否正确配置了类型处理器,确保类型处理器存在、配置正确,并且映射对象属性与数据库字段类型匹配,以解决该错误。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值