技术分享图片 sql (mybatis): 技术分享图片 原因: 经测试,oracle 数据库字段为 Data 型的并不会报错,只有 timestamp 类型会报错。 从后台报错日志中发现(

后台报错:

26-Sep-2018 22:18:08.209 WARNING [http-apr-8080-exec-8] org.springframework.web.servlet.mvc.support.DefaultHandlerExceptionResolver.handleHttpMessageNotWritable Failed to write HTTP message: org.springframework.http.converter.HttpMessageNotWritableException: Could not write JSON: No serializer found for class java.io.ByteArrayInputStream and no properties discovered to create BeanSerializer (to avoid exception, disable SerializationFeature.FAIL_ON_EMPTY_BEANS); nested exception is com.fasterxml.jackson.databind.exc.InvalidDefinitionException: No serializer found for class java.io.ByteArrayInputStream and no properties discovered to create BeanSerializer (to avoid exception, disable SerializationFeature.FAIL_ON_EMPTY_BEANS) (through reference chain: java.util.HashMap[“pageData”]->java.util.ArrayList[2]->java.util.HashMap[“UPDATE_TIME”]->oracle.sql.TIMESTAMP[“stream”

原因:

经测试,oracle 数据库字段为 Data 型的并不会报错,只有 timestamp 类型会报错。

从后台报错日志中发现(through reference chain: java.util.HashMap[“pageData”]->java.util.ArrayList[2]->java.util.HashMap[“UPDATE_TIME”]->oracle.sql.TIMESTAMP[“stream”]) ,

然后发现返回的 map 里面 update_time 字段为 oracle.sql.TIMESTAMP 类型,并不是 java.sql.Timestamp,所以 json 转换出错。

其实都是因为 mybatis 当 ResultMap 为 map 时,会把数据的原始类型原样返回,所以得到的map里面都是 oracle.sql.DATE、oracle.sql.TIMESTAMP 之类的。因为 mybatis 在没有指定类型时都会采用 ObjectTypeHandle 来处理字段。

解决方案:

自定义 typeHandle 来统一处理数据库字段类型为 timestamp 等特殊的字段。

这里 typeHandle 里面使用注解配置 JdbcType 和 JavaType。这两个注解的定义是:

@MappedTypes 定义的是 JavaType 类型,可以指定哪些 Java 类型被拦截。
@MappedJdbcTypes 定义的是 JdbcType 类型,它需要满足枚举类 org.apache.ibatis.type.JdbcType 所列的枚举类型。

代码如下:

myBatis 的配置文件中加入:

<typeHandlers>
    <typeHandler handler="com.yule.system.typehandler.MyObjectTypeHandle"/>
</typeHandlers>

新增新的 java 类:

package com.yule.system.typehandler;

import oracle.sql.DATE;
import oracle.sql.TIMESTAMP;
import oracle.sql.TIMESTAMPLTZ;
import oracle.sql.TIMESTAMPTZ;
import org.apache.ibatis.type.*;

import java.sql.CallableStatement;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.util.Date;

/**

  • 模仿 ObjectTypeHandle 来处理 timestamp 报错问题

  • @author yule

  • @date 2018/9/26 22:43
    */
    @MappedTypes({Object.class})
    @MappedJdbcTypes(value = {JdbcType.TIMESTAMP})
    public class MyObjectTypeHandle extends BaseTypeHandler {

    public MyObjectTypeHandle() {
    }

    @Override
    public void setNonNullParameter(PreparedStatement ps, int i, Object parameter, JdbcType jdbcType) throws SQLException {
    ps.setObject(i, parameter);
    }

    @Override
    public Object getNullableResult(ResultSet rs, String columnName) throws SQLException {
    Object result = rs.getObject(columnName);
    return rs.wasNull() ? null : dealResult(result);
    }

    @Override
    public Object getNullableResult(ResultSet rs, int columnIndex) throws SQLException {
    Object result = rs.getObject(columnIndex);
    return rs.wasNull() ? null : dealResult(result);
    }

    @Override
    public Object getNullableResult(CallableStatement cs, int columnIndex) throws SQLException {
    Object result = cs.getObject(columnIndex);
    return cs.wasNull() ? null : dealResult(result);
    }

    /**

    • 为了解决错误:
    • 26-Sep-2018 14:21:06.634 WARNING [http-apr-8080-exec-6] org.springframework.web.servlet.mvc.support.DefaultHandlerExceptionResolver.handleHttpMessageNotWritable Failed to write HTTP message: org.springframework.http.converter.HttpMessageNotWritableException:
    • Could not write JSON: No serializer found for class java.io.ByteArrayInputStream and no properties discovered to create BeanSerializer
    • (to avoid exception, disable SerializationFeature.FAIL_ON_EMPTY_BEANS);
    • nested exception is com.fasterxml.jackson.databind.exc.InvalidDefinitionException: No serializer found for class java.io.ByteArrayInputStream and no properties discovered to create BeanSerializer (to avoid exception, disable SerializationFeature.FAIL_ON_EMPTY_BEANS)
    • (through reference chain: java.util.HashMap[“pageData”]->java.util.ArrayList[0]->java.util.HashMap[“UPDATE_TIME”]->oracle.sql.TIMESTAMP[“stream”])
    • @param result
    • @return
    • @throws SQLException
      */
      private Object dealResult(Object result) throws SQLException {
      if (result instanceof TIMESTAMP) {
      return new Date(((TIMESTAMP) result).dateValue().getTime());
      } else if (result instanceof DATE) {
      return new Date(((DATE) result).dateValue().getTime());
      } else if (result instanceof TIMESTAMPLTZ) {
      return new Date(((TIMESTAMPLTZ) result).dateValue().getTime());
      } else if (result instanceof TIMESTAMPTZ) {
      return new Date(((TIMESTAMPTZ) result).dateValue().getTime());
      } else{
      return result;
      }
      }
      }
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值