zoukankan      html  css  js  c++  java
  • 解决:oracle+myBatis ResultMap 类型为 map 时返回结果中存在 timestamp 时使用 jackson 转 json 报错

    前言:最近在做一个通用查询单表的组件,所以 sql 的写法就是 select *,然后 resultType="map" ,然后使用 jackson @ResponseBody 返回前端报错。 转载请注明出处:https://www.cnblogs.com/yuxiaole/p/9708485.html

    后台报错:

    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):

    sql (mybatis):

    原因:

    经测试,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<Object> {
    
        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;
            }
        }
    }

    转载请注明出处:https://www.cnblogs.com/yuxiaole/p/9708485.html 

  • 相关阅读:
    [Swift]LeetCode1093. 大样本统计 | Statistics from a Large Sample
    [Swift]鸡尾酒排序 | Cocktail Sort
    [Swift]插入排序 | Insertion sort
    [Swift]求最大公约数和最小公倍数
    [Swift]LeetCode1087. 字母切换 | Permutation of Letters
    [Swift]LeetCode1086. 前五科的均分 | High Five
    [Swift]LeetCode1088. 易混淆数 II | Confusing Number II
    [Swift]LeetCode1090. 受标签影响的最大值 | Largest Values From Labels
    [Swift]LeetCode1091. 二进制矩阵中的最短路径 | Shortest Path in Binary Matrix
    [Swift]LeetCode1092. 最短公共超序列 | Shortest Common Supersequence
  • 原文地址:https://www.cnblogs.com/yuxiaole/p/9708485.html
Copyright © 2011-2022 走看看