typeHandler解决JSONArray类型无法使数据入库的问题
在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类型。