mybatis 自定义转换器 ListTypeHandler

mybatis 自定义转换器 ListTypeHandler
实现存库 list->string

读库 string->list
package com.jeesite.modules.handler;

import com.alibaba.fastjson.JSON;
import com.alibaba.fastjson.TypeReference;
import com.jeesite.common.lang.ObjectUtils;
import com.jeesite.common.lang.StringUtils;
import org.apache.ibatis.type.BaseTypeHandler;
import org.apache.ibatis.type.JdbcType;
import org.apache.ibatis.type.MappedJdbcTypes;
import org.apache.ibatis.type.MappedTypes;
<!--more-->
import java.sql.CallableStatement;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.util.ArrayList;
import java.util.List;
/**
 * @author liuxin
 * @version 1.0
 * @date 2022/8/23 12:25
 */
@MappedJdbcTypes(JdbcType.VARCHAR)
@MappedTypes({List.class})
public abstract class ListTypeHandler<T> extends BaseTypeHandler<List<T>> {

  @Override
  public void setNonNullParameter(PreparedStatement ps, int i, List<T> parameter, JdbcType jdbcType)
      throws SQLException {
    String content = ObjectUtils.isEmpty(parameter) ? null : JSON.toJSONString(parameter);
    ps.setString(i, content);
  }

  @Override
  public List<T> getNullableResult(ResultSet rs, String columnName) throws SQLException {
    return this.getListByJsonArrayString(rs.getString(columnName));
  }

  @Override
  public List<T> getNullableResult(ResultSet rs, int columnIndex) throws SQLException {
    return this.getListByJsonArrayString(rs.getString(columnIndex));
  }

  @Override
  public List<T> getNullableResult(CallableStatement cs, int columnIndex) throws SQLException {
    return this.getListByJsonArrayString(cs.getString(columnIndex));
  }

  private List<T> getListByJsonArrayString(String content) {
    return StringUtils.isBlank(content)
        ? new ArrayList<>()
        : JSON.parseObject(content, this.specificType());
  }

  /**
   * 具体类型,由子类提供
   *
   * @return 具体类型
   */
  protected abstract TypeReference<List<T>> specificType();
}
 Java 泛型 如果在 ListTypeHandler 类中直接提供 TypeReference<List<T>> 这种类型,那就等效于TypeReference<List<Object>> 这种类型,后续 fastjson 在转换时无法确定具体的 Java 类型,转换后的类型最终就会是 List<JSONObject> ;同理,如果使用 Jackson 作为 JSON 转换工具,不确定具体类型时,最总会被转换为LinkedHashMap 类型,都需要再使用 TypeReference 来转换一次。

自定义TypeHandler的使用笔记 可通过自定义的TypeHandler实现某个属性在插入数据库以及查询时的自动转换。
package com.jeesite.modules.handler;

/**
 * @author liuxin
 * @version 1.0
 * @date 2022/8/23 12:33
 */

import com.alibaba.fastjson.TypeReference;
import com.pdd.pop.sdk.http.api.pop.response.PddGoodsListGetResponse;

import java.util.List;

public class SkuListItemSpecDetailsTypeHandler
        extends ListTypeHandler<PddGoodsListGetResponse.GoodsListGetResponseGoodsListItemSkuListItemSpecDetailsItem> {
    @Override
    protected TypeReference<List<PddGoodsListGetResponse.GoodsListGetResponseGoodsListItemSkuListItemSpecDetailsItem>> specificType() {
        return new TypeReference<List<PddGoodsListGetResponse.GoodsListGetResponseGoodsListItemSkuListItemSpecDetailsItem>>() {
        };
    }
}

实体类上直接用注解:

 typeHandler = ListTypeHandler.class,
 javaType = PddGoodsListGetResponse.GoodsListGetResponseGoodsListItemSkuListItemSpecDetailsItem.class

mapper.xml

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.jeesite.modules.pdd.dao.PddGoodsSkuListDao">

    <resultMap type="com.jeesite.modules.pdd.entity.PddGoodsSkuList" id="PddGoodsSkuListMap">
        <!--column需要和数据库字段名对应,当实体上面设置了属性名则和属性名对应-->
        <result column="specDetails" property="specDetails"
                javaType="java.util.List" typeHandler="com.jeesite.modules.handler.SkuListItemSpecDetailsObjectTypeHandler"/>
    </resultMap>
    <select id="findList" resultMap="PddGoodsSkuListMap">
        SELECT ${sqlMap.column.toSql()}
        FROM ${sqlMap.table.toSql()}
        <where>
            ${sqlMap.where.toSql()}
        </where>
        ORDER BY ${sqlMap.order.toSql()}
    </select>
</mapper>