【已解决】Error querying database. Cause_ java.sql.SQLDataException_ ORA-01861_ 文字与格式字符串不匹配

🎉在工作场景中遇到这样一个需求,需要统计指定时间段内网站的访客数据。例如,统计从 2022 年 4 月 1 号至 2022 年 4 月 30 号内网站的所有访客记录,之后生成一个统计报表。需要注意的是,这里的时间范围是指从 2022 年 4 月 1 号凌晨 0 点 0 分 0 秒 至 2022 年 4 月 30 号 23 点 59 分 59 秒,单位必须精确到秒!

🎉对于这样的一个需求,解决思路其实很简单,无非就是对后台数据库进行时间范围的查询操作,编写 SQL 语句代码,肯定要用到 between and. 然而这里有一个细节需要注意,就是在 Oracle 数据库中,虽然 between and包含端点值 的,但是在 Oracle 数据库中使用 to_date 函数对字符串进行转换时,默认的时间筛选是从当天的 00:00:00 开始计算的。也就是说,如果想要查询 2022-04-01 至 2022-04-02 的数据,使用日期函数 to_date 转换时,筛选的是 2022-04-01 00:00:00 至 2022-04-02 00:00:00 的数据,它是不包含 2022-04-02 这天的数据的

🎉为了解决这个问题,通常的做法是对于查询的结束日期采取加1的操作,这样自然就包含了结束日期当天的数据了。在我所负责维护的这个项目中,前任维护者也是采取了这样的方法,但是在后期应用于生产环境中,出现这样的问题。当指定的时间范围包含了每个月最后一天时,统计数据便出现了 bug. 反馈到我这边之后,我通过查询后台错误日志,定位到这样的错误提示:

java 复制代码
org.springframework.dao.DataIntegrityViolationException: 
### Error querying database.  Cause: java.sql.SQLDataException: ORA-01861: 文字与格式字符串不匹配

🎉错误日志提示指出,在执行 SQL 查询时,出现 文字与格式字符串不匹配错误。于是,我便又去扒出后端 SQL 语句编写的代码,下面是出错的代码位置

sql 复制代码
where ADDTIME between to_date(#{beg}, 'yyyy-mm-dd hh24:mi:ss') and to_date(#{end}+1, 'yyyy-mm-dd hh24:mi:ss')

🎉注意到 to_date(#{end}+1 这句代码,end 表示传入的时间字符串,它是形如如 20220430 这样的字符串格式,{end}+1 是为了查询出 20220430 当天的数据。但是,仔细想一下当 20220430+1 此时不就是 20220431 了吗?但是四月不可能有 31 号的,因此联想到每次查询每个月最后一天的数据时总是报错的问题,也就找到了问题的根源了!

🎉因此,{end}+1 这样的操作时不能再继续使用了,必须删掉 +1 这个问题根源。那么,删掉 +1 之后,还能查询出边界日期的当天数据吗?在经过一番了解之后,可以通过修改 Oracle 中 to_date 函数的默认筛选时间的方式解决,可以将默认筛选时间指定为 23:59:59,这样便可以查询出边界日期的当天数据了。

后台业务逻辑代码修改为

java 复制代码
List<ReportMobileVO> pvList = userLoginMapper.getPV2(beg, end + " 23:59:59");

后台 SQL 代码修改为

sql 复制代码
where ADDTIME between to_date(#{beg}, 'yyyy-mm-dd hh24:mi:ss') and to_date(#{end}, 'yyyy-mm-dd hh24:mi:ss')

🎉这里的 end 表示边界日期,在调用接口方法执行 SQL 查询之前,传入的 end 参数拼接字符串 23:59:59,这样可以指定筛选时间。例如,指定之后的筛选时间为 20220430 23:59:59,这样既可以将边界日期当天数据包含在内,又不会出错了,问题得到解决!

🚨最后,需要说明的是,如果错误原因并不是上述所描述的情况,则应该重点仔细检查传入的时间字符串格式。例如,传入的参数是不是形如 20220401这样的格式,如果不是的话,是不能被解析的

相关推荐
源代码•宸7 分钟前
GoLang八股(Go语言基础)
开发语言·后端·golang·map·defer·recover·panic
czlczl200209258 分钟前
OAuth 2.0 解析:后端开发者视角的原理与流程讲解
java·spring boot·后端
颜淡慕潇16 分钟前
Spring Boot 3.3.x、3.4.x、3.5.x 深度对比与演进分析
java·后端·架构
布列瑟农的星空16 分钟前
WebAssembly入门(一)——Emscripten
前端·后端
小突突突2 小时前
Spring框架中的单例bean是线程安全的吗?
java·后端·spring
iso少年2 小时前
Go 语言并发编程核心与用法
开发语言·后端·golang
掘金码甲哥2 小时前
云原生算力平台的架构解读
后端
码事漫谈2 小时前
智谱AI从清华实验室到“全球大模型第一股”的六年征程
后端
码事漫谈2 小时前
现代软件开发中常用架构的系统梳理与实践指南
后端
Mr.Entropy2 小时前
JdbcTemplate 性能好,但 Hibernate 生产力高。 如何选择?
java·后端·hibernate