Mybatis Error attempting to get column xx. Bad format for number '2018-09-18' in column 20(数据类型转换错误)

本文主要是介绍Mybatis Error attempting to get column xx. Bad format for number '2018-09-18' in column 20(数据类型转换错误),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

本文主要分享下类似问题的排查过程。

执行查询时报错,错误日志摘要:

org.springframework.dao.TransientDataAccessResourceException: 
Error attempting to get column 'create_time' from result set. Cause: java.sql.SQLException: Bad format for number '2018-09-18 19:29:02.0' in column 20.

日志提醒的很明显,列'create_time'取值失败,原因是格式不对、不是数字。
那么问题来了:我的MySQL表中,create_timedatetime类型,讲道理应该转换为java.util.Date对象,怎么会转数字呢?

排查

完整日志如下:

Caused by: java.sql.SQLException: Bad format for number '2018-09-18 19:29:02.0' in column 20.at com.mysql.jdbc.SQLError.createSQLException(SQLError.java:965)at com.mysql.jdbc.SQLError.createSQLException(SQLError.java:898)at com.mysql.jdbc.SQLError.createSQLException(SQLError.java:887)at com.mysql.jdbc.SQLError.createSQLException(SQLError.java:861)at com.mysql.jdbc.ResultSetImpl.getDoubleInternal(ResultSetImpl.java:2333)at com.mysql.jdbc.ResultSetImpl.getDoubleInternal(ResultSetImpl.java:2273)at com.mysql.jdbc.ResultSetImpl.getDouble(ResultSetImpl.java:2241)at com.mysql.jdbc.ResultSetImpl.getDouble(ResultSetImpl.java:2253)at org.apache.commons.dbcp2.DelegatingResultSet.getDouble(DelegatingResultSet.java:295)at org.apache.commons.dbcp2.DelegatingResultSet.getDouble(DelegatingResultSet.java:295)at org.dommons.db.jdbc.EssentialResultSet.getDouble(EssentialResultSet.java:216)at org.dommons.db.jdbc.EssentialResultSet.getDouble(EssentialResultSet.java:216)at org.apache.ibatis.type.DoubleTypeHandler.getNullableResult(DoubleTypeHandler.java:37)at org.apache.ibatis.type.DoubleTypeHandler.getNullableResult(DoubleTypeHandler.java:26)at org.apache.ibatis.type.BaseTypeHandler.getResult(BaseTypeHandler.java:66)at org.apache.ibatis.executor.resultset.DefaultResultSetHandler.createRowKeyForMappedProperties(DefaultResultSetHandler.java:1073)at org.apache.ibatis.executor.resultset.DefaultResultSetHandler.createRowKey(DefaultResultSetHandler.java:1030)at org.apache.ibatis.executor.resultset.DefaultResultSetHandler.applyNestedResultMappings(DefaultResultSetHandler.java:957)at org.apache.ibatis.executor.resultset.DefaultResultSetHandler.getRowValue(DefaultResultSetHandler.java:918)at org.apache.ibatis.executor.resultset.DefaultResultSetHandler.handleRowValuesForNestedResultMap(DefaultResultSetHandler.java:881)at org.apache.ibatis.executor.resultset.DefaultResultSetHandler.handleRowValues(DefaultResultSetHandler.java:328)at org.apache.ibatis.executor.resultset.DefaultResultSetHandler.handleResultSet(DefaultResultSetHandler.java:303)at org.apache.ibatis.executor.resultset.DefaultResultSetHandler.handleResultSets(DefaultResultSetHandler.java:196)at org.apache.ibatis.executor.statement.PreparedStatementHandler.query(PreparedStatementHandler.java:64)at org.apache.ibatis.executor.statement.RoutingStatementHandler.query(RoutingStatementHandler.java:79)at org.apache.ibatis.executor.SimpleExecutor.doQuery(SimpleExecutor.java:63)at org.apache.ibatis.executor.BaseExecutor.queryFromDatabase(BaseExecutor.java:326)at org.apache.ibatis.executor.BaseExecutor.query(BaseExecutor.java:156)at org.apache.ibatis.executor.CachingExecutor.query(CachingExecutor.java:109)at org.apache.ibatis.executor.CachingExecutor.query(CachingExecutor.java:83)at org.apache.ibatis.session.defaults.DefaultSqlSession.selectList(DefaultSqlSession.java:148)at org.apache.ibatis.session.defaults.DefaultSqlSession.selectList(DefaultSqlSession.java:141)at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)at java.lang.reflect.Method.invoke(Method.java:498)at org.mybatis.spring.SqlSessionTemplate$SqlSessionInterceptor.invoke(SqlSessionTemplate.java:433)

通过堆栈可以看到,Mybatis使用了DoubleTypeHandler转换了我这个create_time日期数据,用了这个Handler报错是Bad format for number就可以理解了。

那么问题变成为什么Mybatis选择了这个Handler

有个小插曲,最开始没用日志,而是主观的去判断了下问题想走走捷径:

  1. 因为这个值刚好在association中定义的,我怀疑这个标签有问题,会选择错类型转换器。还去除了association重试了下,果然没有错误了。于是莫名的开始嗨,感觉发现了惊天大咪咪。
  2. 有了1的判断,尝试在xml中显式配置jdbcTypeDATE(常见的jdbcType),重试时报错仍然存在。这个时候网上查了查没有类似资料,线索断了。只能回到日志排查源码了。

最终证明我是浪费了时间。 遇到问题还是不能太主观,有明显的日志了,还是按图索骥比较靠谱。

开启debug,F5按调用一层一层跟进去,找到异常拦截和处理的代码:org.apache.ibatis.type.BaseTypeHandler<T>.getResult(ResultSet rs, String columnName)
在这里插入图片描述
看到这段代码,可以注意到两个关键点:

  1. columnName是一个入参,也就是解析结果集可能是遍历时传入的,日志里明确提示了是create_time,应该没问题;

    Error attempting to get column create_time from result set.

  2. 回想最开始我们看到的日志,Cause后面的部分(Bad format for number ‘2018-09-18’ in column 20)是从接收的异常中拿到的。那么接下来应该进入到异常抛出的地方。
  3. 继续调试,终于找到了异常抛出类,并且发现column 20是通过findColumn获取到的; 查看sql结果集的列,发现列名、序号是对的上的。
    在这里插入图片描述

在这里我突然灵光一闪,幡然醒悟了一个思维误区:

日志中提示的一直是column create_timecolumn 20,但并没有说是在转换ResultMapcreate_time属性。也就是说理论上,ResultSet的create_time属性值,可能正被提取解析成ResultMap的任意属性。要验证这一点,直接去MybatisResultMap配置中,查看是否有多个属性的column都配置了create_time

事实证明确实如此,有个金额属性,column映射到了create_time,所以转换报错了。
在这里插入图片描述

总结

这个问题有几个关键点,如果能早些注意、或者有知识储备,解决问题会很快:

  1. column 20com.mysql.jdbc.ResultSetImpl.findColumn找到的列索引。注意,是索引,不是ResultMap中属性的顺序号,所以不要看到这个属性名,就去看这个属性类型转换有什么问题。而应该去查一查执行的sql结果集,第20列是什么内容,ResultMap中是否有多个字段的column错误配置了该列、导致类型不匹配。
  2. Mybatis选择哪个TypeHandler,默认是根据ResultMapproperty的类型来的(当然还可以显示指定)。所以如果发现TypeHandler和预想的不一样,应该意识到问题出在某个propertycolumn不匹配。

又是一个巨大的乌龙,一星期就遇到了两次,而且一次比一次离(di)奇(ji),这次已经超越MySQL ERROR 16 (42000):FUNCTION sum does not exist. Check the ‘Function Name Parsing and Resolution’,成为2019悲惨事件top1。感觉今年剩下的四个月很难熬。

解决问题过程中发现的其他类似事件,也比较有价值:https://blog.csdn.net/m0_43452671/article/details/89315225、https://blog.csdn.net/gugou123/article/details/80920431、

因为错怪了association,特意找了两篇文章再学习了下:https://www.cnblogs.com/yuan951/p/7594176.html、https://www.cnblogs.com/duanxz/p/3830509.html

这篇关于Mybatis Error attempting to get column xx. Bad format for number '2018-09-18' in column 20(数据类型转换错误)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/358900

相关文章

MyBatis常用XML语法详解

《MyBatis常用XML语法详解》文章介绍了MyBatis常用XML语法,包括结果映射、查询语句、插入语句、更新语句、删除语句、动态SQL标签以及ehcache.xml文件的使用,感兴趣的朋友跟随小... 目录1、定义结果映射2、查询语句3、插入语句4、更新语句5、删除语句6、动态 SQL 标签7、ehc

MyBatis延迟加载与多级缓存全解析

《MyBatis延迟加载与多级缓存全解析》文章介绍MyBatis的延迟加载与多级缓存机制,延迟加载按需加载关联数据提升性能,一级缓存会话级默认开启,二级缓存工厂级支持跨会话共享,增删改操作会清空对应缓... 目录MyBATis延迟加载策略一对多示例一对多示例MyBatis框架的缓存一级缓存二级缓存MyBat

Linux下利用select实现串口数据读取过程

《Linux下利用select实现串口数据读取过程》文章介绍Linux中使用select、poll或epoll实现串口数据读取,通过I/O多路复用机制在数据到达时触发读取,避免持续轮询,示例代码展示设... 目录示例代码(使用select实现)代码解释总结在 linux 系统里,我们可以借助 select、

mybatis直接执行完整sql及踩坑解决

《mybatis直接执行完整sql及踩坑解决》MyBatis可通过select标签执行动态SQL,DQL用ListLinkedHashMap接收结果,DML用int处理,注意防御SQL注入,优先使用#... 目录myBATiFBNZQs直接执行完整sql及踩坑select语句采用count、insert、u

MyBatis Plus大数据量查询慢原因分析及解决

《MyBatisPlus大数据量查询慢原因分析及解决》大数据量查询慢常因全表扫描、分页不当、索引缺失、内存占用高及ORM开销,优化措施包括分页查询、流式读取、SQL优化、批处理、多数据源、结果集二次... 目录大数据量查询慢的常见原因优化方案高级方案配置调优监控与诊断总结大数据量查询慢的常见原因MyBAT

MyBatis/MyBatis-Plus同事务循环调用存储过程获取主键重复问题分析及解决

《MyBatis/MyBatis-Plus同事务循环调用存储过程获取主键重复问题分析及解决》MyBatis默认开启一级缓存,同一事务中循环调用查询方法时会重复使用缓存数据,导致获取的序列主键值均为1,... 目录问题原因解决办法如果是存储过程总结问题myBATis有如下代码获取序列作为主键IdMappe

C#使用iText获取PDF的trailer数据的代码示例

《C#使用iText获取PDF的trailer数据的代码示例》开发程序debug的时候,看到了PDF有个trailer数据,挺有意思,于是考虑用代码把它读出来,那么就用到我们常用的iText框架了,所... 目录引言iText 核心概念C# 代码示例步骤 1: 确保已安装 iText步骤 2: C# 代码程

Pandas处理缺失数据的方式汇总

《Pandas处理缺失数据的方式汇总》许多教程中的数据与现实世界中的数据有很大不同,现实世界中的数据很少是干净且同质的,本文我们将讨论处理缺失数据的一些常规注意事项,了解Pandas如何表示缺失数据,... 目录缺失数据约定的权衡Pandas 中的缺失数据None 作为哨兵值NaN:缺失的数值数据Panda

C++中处理文本数据char与string的终极对比指南

《C++中处理文本数据char与string的终极对比指南》在C++编程中char和string是两种用于处理字符数据的类型,但它们在使用方式和功能上有显著的不同,:本文主要介绍C++中处理文本数... 目录1. 基本定义与本质2. 内存管理3. 操作与功能4. 性能特点5. 使用场景6. 相互转换核心区别

python库pydantic数据验证和设置管理库的用途

《python库pydantic数据验证和设置管理库的用途》pydantic是一个用于数据验证和设置管理的Python库,它主要利用Python类型注解来定义数据模型的结构和验证规则,本文给大家介绍p... 目录主要特点和用途:Field数值验证参数总结pydantic 是一个让你能够 confidentl