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


相关文章
|
3月前
|
SQL JSON Java
告别字符串拼接:用Java文本块优雅处理多行字符串
告别字符串拼接:用Java文本块优雅处理多行字符串
399 108
|
5月前
|
SQL JSON Java
告别拼接噩梦:Java文本块让多行字符串更优雅
告别拼接噩梦:Java文本块让多行字符串更优雅
567 82
|
9月前
|
SQL 自然语言处理 数据库
【Azure Developer】分享两段Python代码处理表格(CSV格式)数据 : 根据每列的内容生成SQL语句
本文介绍了使用Python Pandas处理数据收集任务中格式不统一的问题。针对两种情况:服务名对应多人拥有状态(1/0表示),以及服务名与人名重复列的情况,分别采用双层for循环和字典数据结构实现数据转换,最终生成Name对应的Services列表(逗号分隔)。此方法高效解决大量数据的人工处理难题,减少错误并提升效率。文中附带代码示例及执行结果截图,便于理解和实践。
241 4
|
5月前
|
自然语言处理 Java Apache
在Java中将String字符串转换为算术表达式并计算
具体的实现逻辑需要填写在 `Tokenizer`和 `ExpressionParser`类中,这里只提供了大概的框架。在实际实现时 `Tokenizer`应该提供分词逻辑,把输入的字符串转换成Token序列。而 `ExpressionParser`应当通过递归下降的方式依次解析
357 14
|
9月前
|
存储 缓存 安全
Java 字符串详解
本文介绍了 Java 中的三种字符串类型:String、StringBuffer 和 StringBuilder,详细讲解了它们的区别与使用场景。String 是不可变的字符串常量,线程安全但操作效率较低;StringBuffer 是可变的字符串缓冲区,线程安全但性能稍逊;StringBuilder 同样是可变的字符串缓冲区,但非线程安全,性能更高。文章还列举了三者的常用方法,并总结了它们在不同环境下的适用情况及执行速度对比。
210 17
|
9月前
|
存储 缓存 安全
Java字符串缓冲区
字符串缓冲区是用于处理可变字符串的容器,Java中提供了`StringBuffer`和`StringBuilder`两种实现。由于`String`类不可变,当需要频繁修改字符串时,使用缓冲区更高效。`StringBuffer`是一个线程安全的容器,支持动态扩展、任意类型数据转为字符串存储,并提供多种操作方法(如`append`、`insert`、`delete`等)。通过这些方法,可以方便地对字符串进行添加、插入、删除等操作,最终将结果转换为字符串。示例代码展示了如何创建缓冲区对象并调用相关方法完成字符串操作。
255 13
|
SQL Java 索引
java小工具util系列2:字符串工具
java小工具util系列2:字符串工具
313 83
|
存储 安全 Java
Java零基础-字符串详解
【10月更文挑战第18天】Java零基础教学篇,手把手实践教学!
238 60
|
Java 数据库
java小工具util系列1:日期和字符串转换工具
java小工具util系列1:日期和字符串转换工具
225 26