Mybatis实现自定义转换器(清晰附实例版)

场景

比如:用户的状态:0为可用,1为被禁止,-1为被删除。

  1. 数据库存储的是整形。

    这样的:

在这里插入图片描述

  1. 如果要代码的可读性,我们可以使用Enum或者单独一个类来实现。我本身采用的是Enum的方式来实现的,不优雅,但是还可以吧。

    /**
     * @author 青韵
     * @date 2020/10/12 - 20:28
     */
    public enum AccountEnableTypeEnum {
        /**
         * 正常使用
         */
        OK((byte) 1),
        /**
         * 被禁止了
         */
        FORBIDDEN((byte) 0),
        /**
         * 用户已注销
         */
        DELETE((byte) -1);
    
        private Byte value;
    
        AccountEnableTypeEnum(Byte value) {
            this.value = value;
        }
    
        public Byte getValue() {
            return value;
        }
    
        /**
        *  在做类型转换可以用上
        **/
        public static AccountEnableTypeEnum getInstance(Byte value) {
            for (AccountEnableTypeEnum accountEnableTypeEnum : AccountEnableTypeEnum.values()) {
                if (accountEnableTypeEnum.getValue().equals(value)) {
                    return accountEnableTypeEnum;
                }
            }
    
            return null;
        }
    }
    
  2. 由于Enum默认采用的是名字的方式来实现的,所以我们存储的(-1, 0, 1)无法转换为相应的对象。

    public class EnumTypeHandler<E extends Enum<E>> extends BaseTypeHandler<E> {
      @Override
      public E getNullableResult(ResultSet rs, String columnName) throws SQLException {
        String s = rs.getString(columnName);
        return s == null ? null : Enum.valueOf(type, s);
      }
    }
    

    这里我们有两个办法:

    1. 我们知道:**枚举类型会有顺序的,通过.ordinal()**来获取,例如:AccountEnableTypeEnum.DELETE.ordinal()就是2

    2. 由于我们设置了1对应可以使用,0表示被禁止,-1表示账户已经删除。而且我们设置了value。

      所以我们可以 自定义转换器,淦他

要实现自定义的转换器,可以

先实现TypeHandler接口或者继承BaseTypeHandler等。
实现TypeHandler接口,实现内部的方法:
public interface TypeHandler<T> {
  /**
  *  设置参数时用,表示将Java类型转换为对应的数据库类型
  **/ 
  void setParameter(PreparedStatement ps, int i, T parameter, JdbcType jdbcType) throws SQLException;

  /**
  *  下面三个都是:从数据库获取数据时,转换为相应的对象时的方法
  **/
  T getResult(ResultSet rs, String columnName) throws SQLException;

  T getResult(ResultSet rs, int columnIndex) throws SQLException;

  T getResult(CallableStatement cs, int columnIndex) throws SQLException;
}
继承BaseTypeHandler

官方帮我们写好了一些流程,按照思路帮我们实现了那几个接口,以及对于异常的处理,把需要实现的具体定义为了对应的方法,只需要处理转换的部分即可,何乐而不为呢?

public abstract class BaseTypeHandler<T> extends TypeReference<T> implements TypeHandler<T> {
  @Override
  public void setParameter(PreparedStatement ps, int i, T parameter, JdbcType jdbcType) throws SQLException {
    if (parameter == null) {
      if (jdbcType == null) {
        throw new TypeException("JDBC requires that the JdbcType must be specified for all nullable parameters.");
      }
      try {
        ps.setNull(i, jdbcType.TYPE_CODE);
      } catch (SQLException e) {
        throw new TypeException("Error setting null for parameter #" + i + " with JdbcType " + jdbcType + " . "
              + "Try setting a different JdbcType for this parameter or a different jdbcTypeForNull configuration property. "
              + "Cause: " + e, e);
      }
    } else {
      try {
        setNonNullParameter(ps, i, parameter, jdbcType);
      } catch (Exception e) {
        throw new TypeException("Error setting non null for parameter #" + i + " with JdbcType " + jdbcType + " . "
              + "Try setting a different JdbcType for this parameter or a different configuration property. "
              + "Cause: " + e, e);
      }
    }
  }

  @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);
    }
  }

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

  @Override
  public T getResult(CallableStatement cs, int columnIndex) throws SQLException {
    try {
      return getNullableResult(cs, columnIndex);
    } catch (Exception e) {
      throw new ResultMapException("Error attempting to get column #" + columnIndex + " from callable statement.  Cause: " + e, e);
    }
  }

  /**
  *  实现下列这些就可以了
  **/
  public abstract void setNonNullParameter(PreparedStatement ps, int i, T parameter, JdbcType jdbcType) throws SQLException;

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

  public abstract T getNullableResult(ResultSet rs, int columnIndex) throws SQLException;

  public abstract T getNullableResult(CallableStatement cs, int columnIndex) throws SQLException;

}
我是这么做的
public class AccountTypeEnumHandler extends BaseTypeHandler<AccountEnableTypeEnum> {
    @Override
    public void setNonNullParameter(PreparedStatement ps, int i, AccountEnableTypeEnum parameter, JdbcType jdbcType) throws SQLException {
        ps.setByte(i, parameter.getValue());
    }

    @Override
    public AccountEnableTypeEnum getNullableResult(ResultSet rs, String columnName) throws SQLException {
        return AccountEnableTypeEnum.getInstance(rs.getByte(columnName));
    }

    @Override
    public AccountEnableTypeEnum getNullableResult(ResultSet rs, int columnIndex) throws SQLException {
        return AccountEnableTypeEnum.getInstance(rs.getByte(columnIndex));
    }

    @Override
    public AccountEnableTypeEnum getNullableResult(CallableStatement cs, int columnIndex) throws SQLException {
        return AccountEnableTypeEnum.getInstance(cs.getByte(columnIndex));
    }
}
接下来就是配置,总体来说有两种方式,我个人推荐第一种,因为有提示,不易出错。
两种方式都需要先在mybatis的配置文件去配置类型转换器。

ps:类型转换器是你自己的,我这块写的是我的类型转换器,不代表是你的

在这里插入图片描述

方式一:在类上加注解
  1. MappedJdbcTypes:表示数据库对应的类型是什么,填你自己的
  2. MappedTypes:表示要处理的类型转换是哪个,填你自己的

在这里插入图片描述

方式二:在xml配置文件中加上两个参数

在这里插入图片描述

恭喜,大功告成

后记

如果你想要的是局部改变,比如对于一个mapper,想做单独的转换,那么可以这样。

在这里插入图片描述

OK,大功告成
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
要将自定义类型转换为JSON集合,可以按照以下步骤操作: 1. 创建一个MyBatis TypeHandler实现类,实现TypeHandler接口,或者继承BaseTypeHandler类。例如,可以创建一个名为JsonListTypeHandler的类。 2. 在TypeHandler实现类中重写setNonNullParameter方法和getNullableResult方法。setNonNullParameter方法用于将Java对象转换为数据库字段值,getNullableResult方法用于将数据库字段值转换为Java对象。 3. 在setNonNullParameter方法中,可以使用JSON库(如Jackson或GSON)将Java对象转换为JSON字符串,然后将JSON字符串设置到PreparedStatement对象中。 4. 在getNullableResult方法中,可以从ResultSet对象中获取数据库字段值,并使用JSON库将JSON字符串转换为Java对象。 5. 在MyBatis配置文件(如mybatis-config.xml)中注册这个TypeHandler。可以通过typeHandlers标签将TypeHandler实现类与自定义的Java类型关联起来。例如,可以添加以下配置: ```xml <typeHandlers> <typeHandler handler="com.example.JsonListTypeHandler" javaType="com.example.CustomTypeList"/> </typeHandlers> ``` 这里的"com.example.JsonListTypeHandler"是TypeHandler实现类的完整类名,"com.example.CustomTypeList"是自定义Java类型的完整类名。 6. 在映射文件中,使用自定义Java类型和对应的数据库字段。例如: ```xml <resultMap id="customTypeListMap" type="com.example.CustomTypeList"> <id property="id" column="id" /> <result property="name" column="name" /> </resultMap> ``` 通过以上步骤,就可以将自定义类型转换为JSON集合了。在查询时,MyBatis将自动应用TypeHandler,将数据库字段值转换为Java对象;在插入或更新时,MyBatis将自动应用TypeHandler,将Java对象转换为数据库字段值。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值