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

简介: 【已解决】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. 反馈到我这边之后,我通过查询后台错误日志,定位到这样的错误提示:

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

🎉错误日志提示指出,在执行 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,这样便可以查询出边界日期的当天数据了。

后台业务逻辑代码修改为

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

后台 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这样的格式,如果不是的话,是不能被解析的。


相关文章
|
27天前
|
SQL Java 索引
java小工具util系列2:字符串工具
java小工具util系列2:字符串工具
138 83
|
1月前
|
存储 安全 Java
Java零基础-字符串详解
【10月更文挑战第18天】Java零基础教学篇,手把手实践教学!
108 60
|
27天前
|
Java 数据库
java小工具util系列1:日期和字符串转换工具
java小工具util系列1:日期和字符串转换工具
54 26
|
1月前
|
存储 缓存 安全
java 中操作字符串都有哪些类,它们之间有什么区别
Java中操作字符串的类主要有String、StringBuilder和StringBuffer。String是不可变的,每次操作都会生成新对象;StringBuilder和StringBuffer都是可变的,但StringBuilder是非线程安全的,而StringBuffer是线程安全的,因此性能略低。
46 8
|
1月前
|
缓存 算法 Java
本文聚焦于Java内存管理与调优,介绍Java内存模型、内存泄漏检测与预防、高效字符串拼接、数据结构优化及垃圾回收机制
在现代软件开发中,性能优化至关重要。本文聚焦于Java内存管理与调优,介绍Java内存模型、内存泄漏检测与预防、高效字符串拼接、数据结构优化及垃圾回收机制。通过调整垃圾回收器参数、优化堆大小与布局、使用对象池和缓存技术,开发者可显著提升应用性能和稳定性。
49 6
|
28天前
|
Java
Java将OffsetDateTime格式化为 yyyy-MM-dd HH:mm:ss 如何写代码?
Java将OffsetDateTime格式化为 yyyy-MM-dd HH:mm:ss 如何写代码?
31 0
|
2月前
|
JSON 前端开发 Java
震惊!图文并茂——Java后端如何响应不同格式的数据给前端(带源码)
文章介绍了Java后端如何使用Spring Boot框架响应不同格式的数据给前端,包括返回静态页面、数据、HTML代码片段、JSON对象、设置状态码和响应的Header。
163 1
震惊!图文并茂——Java后端如何响应不同格式的数据给前端(带源码)
|
2月前
|
Java 数据库
案例一:去掉数据库某列中的所有英文,利用java正则表达式去做,核心:去掉字符串中的英文
这篇文章介绍了如何使用Java正则表达式从数据库某列中去除所有英文字符。
58 15
|
2月前
|
Java
让星星⭐月亮告诉你,Java异常分类[Throwable(Error/Exception(RuntimeException/其他异常)) 检查时异常 非检查时异常]
本文深入解析了Java异常处理机制,重点介绍了`Throwable`类及其子类`Error`和`Exception`,并通过实例代码、流程图和表格详细解释了异常的分类、区别及处理方法,帮助读者掌握异常处理的关键技巧,提升程序的稳定性和健壮性。
65 1
|
2月前
|
Java
JAVA易错点详解(数据类型转换、字符串与运算符)
JAVA易错点详解(数据类型转换、字符串与运算符)
59 4