typeHandler解决JSONArray类型无法使数据入库的问题

发布时间 2023-07-28 18:38:11作者: 山茶花llia

  在java中,一个对象类往往会有很多用途,比如用于接口请求时的传参,或mapper入库时的传参,但请求接口需要的数据类型可能不能用于mapper插入数据库表。
  例如在请求接口时,需要传递一个数组类型的字段,但数据库只能插入varchar类型,需要将数组类型的字段转为字符串类型。
  可用typeHandler(类型转换器)实现一个对象类既适用于接口传参,又可以用于插入数据。

1.在字段中加入注解@ColumnType

@Data
public class test {
@ColumnType(jdbcType = JdbcType.VARCHAR,typeHandler = JsonArrayToStringHandler.class)
private JSONArray locs;
}

2.JsonArrayToStringHandler.class的具体实现

@Component
public class JsonArrayToStringHandler extends BaseTypeHandler<JSONArray> {


    @Override
    public void setNonNullParameter(PreparedStatement ps, int i, JSONArray parameter, JdbcType jdbcType) throws SQLException {
        ps.setString(i, JSONUtil.toJsonStr(parameter));
    }

    @Override
    public JSONArray getNullableResult(ResultSet rs, String columnName) throws SQLException {
        String string = rs.getString(columnName);
        return JSONObject.parseArray(string);
    }

    @Override
    public JSONArray getNullableResult(ResultSet rs, int columnIndex) throws SQLException {
        String string = rs.getString(columnIndex);
        return JSONObject.parseArray(string);
    }

    @Override
    public JSONArray getNullableResult(CallableStatement cs, int columnIndex) throws SQLException {
        String string = cs.getString(columnIndex);
        return JSONObject.parseArray(string);
    }
}

3.插入数据

public interface TestMapper extends CommonRepository<test> {

}
public class practis implements IPractisService {
  @Autowired
  private TestMapper testMapper ;

  @Override
  public void eventPublish(Test test) {
      testMapper.insert(test);
    }
}

  若使用xml实现插入,则可以如下写:

<insert id="insertPerson" parameterType="person">
        INSERT INTO person (id,name,sex,hobbys,data_time,locs) values(#{id},#{name},#{sex},#{locs,typeHandler=com.test.typeHandler.JsonArrayToStringHandler})
</insert>

4.TypeHandler概念

  TypeHandler,类型转换器,在mybatis中用于实现java类型和JDBC类型的相互转换;
  mybatis使用prepareStatement来进行参数设置的时候,需要通过typeHandler将传入的java参数设置成合适的jdbc类型参数,这个过程实际上是通过调用PrepareStatement不同的set方法实现的;在获取结果返回之后,也需要将返回的结果转换成我们需要的java类型,这时候是通过调用ResultSet对象不同类型的get方法时间的;所以不同类型的typeHandler其实就是调用PrepareStatement和ResultSet的不同方法来进行类型的转换,有些时候会在调用PrepareStatement和ResultSet的相关方法之前,可以对传入的参数进行一定的处理。
  当我们没有指定typeHandler的时候mybatis会根据传入参数的类型和返回值的类型调用默认的typeHandler进行处理;对于一个typeHandler需要配置java类型(javaType)和JDBC类型(jdbcType),typeHandler的作用就是实现这两种类型的转换,在传入的参数为指定的Java类型时,将其转换为指定的JDBC类型,当返回值为指定JDBC类型时将其转换为配置的Java类型。