Oracle 中关于 Between and 日期边界问题

简介: 之所以会写日期边界范围的问题,是因为我在查询日期的时候,刚开始数据都能正常获取,但是突然有些数据获取不到。经过一番折腾发现, Oracle 在对日期格式的字段进行查询时,会对字符串进行了隐式转换。 以下就是错误的查询语句 -- create_time 的时间为 2019/3/26 23:00:00 SELECT * FROM T_CLASS T WHERE T.

之所以会写日期边界范围的问题,是因为我在查询日期的时候,刚开始数据都能正常获取,但是突然有些数据获取不到。刚开始以为是 Between and 的边界问题。但是经过一番折腾发现,发现是 Oracle 在对日期格式的字段进行查询时,会对字符串进行了隐式转换。

以下就是错误的查询语句

-- create_time 的时间为 2019/3/26 23:00:00
SELECT *
  FROM T_CLASS T
 WHERE T.CREATE_TIME BETWEEN '2019-03-01' AND '2019-03-26';

一、结论

  1. Oracle 对字符串进行日期转换时,如果是年月日的形式,即“2019-03-26”,将会被转为“2019-03-26 00:00:00”
  2. Oracle 在进行日期比较时,最好直接指定日期比较的格式,不要进行隐式转换

二、初始化数据

  1. 初始化表结构
create table T_CLASS
(
  class_id    NUMBER not null,
  class_name  VARCHAR2(100),
  create_time DATE
)
  1. 初始化表数据
insert into T_CLASS (CLASS_ID, CLASS_NAME, CREATE_TIME)
values (1, '一班', to_date('26-03-2019 23:00:00', 'dd-mm-yyyy hh24:mi:ss'));

insert into T_CLASS (CLASS_ID, CLASS_NAME, CREATE_TIME)
values (2, '二班', to_date('26-03-2019 23:00:00', 'dd-mm-yyyy hh24:mi:ss'));

insert into T_CLASS (CLASS_ID, CLASS_NAME, CREATE_TIME)
values (3, '三班', to_date('26-03-2019 23:00:00', 'dd-mm-yyyy hh24:mi:ss'));

insert into T_CLASS (CLASS_ID, CLASS_NAME, CREATE_TIME)
values (4, '四班', to_date('26-03-2019 23:00:00', 'dd-mm-yyyy hh24:mi:ss'));

insert into T_CLASS (CLASS_ID, CLASS_NAME, CREATE_TIME)
values (5, '五班', to_date('26-03-2019 23:00:00', 'dd-mm-yyyy hh24:mi:ss'));
  1. 查询结果
CLASS_ID CLASS_NAME CREATE_TIME
1 一班 2019/3/26 23:00:00
2 二班 2019/3/26 23:00:00
3 三班 2019/3/26 23:00:00
4 四班 2019/3/26 23:00:00
5 五班 2019/3/26 23:00:00

三、日期查询的几种形式

1.日期直接比较

查询语句

--隐式转换
SELECT * FROM T_CLASS T WHERE T.CREATE_TIME = '2019-03-26';

查询结果

CLASS_ID CLASS_NAME CREATE_TIME

2.日期转换为字符串后比较

查询语句

--显示转换
SELECT *
  FROM T_CLASS T
 WHERE TO_CHAR(T.CREATE_TIME, 'YYYY-MM-DD') = '2019-03-26';

查询结果

CLASS_ID CLASS_NAME CREATE_TIME
1 一班 2019/3/26 23:00:00
2 二班 2019/3/26 23:00:00
3 三班 2019/3/26 23:00:00
4 四班 2019/3/26 23:00:00
5 五班 2019/3/26 23:00:00

3.日期范围比较(边界值)

查询语句

--隐式转换
SELECT *
  FROM T_CLASS T
 WHERE T.CREATE_TIME BETWEEN '2019-03-01' AND '2019-03-26';

查询结果

CLASS_ID CLASS_NAME CREATE_TIME

4.日期范围比较(超过边界值)

--隐式转换
SELECT *
  FROM T_CLASS T
 WHERE T.CREATE_TIME BETWEEN '2019-03-01' AND '2019-03-27';

查询结果

CLASS_ID CLASS_NAME CREATE_TIME
1 一班 2019/3/26 23:00:00
2 二班 2019/3/26 23:00:00
3 三班 2019/3/26 23:00:00
4 四班 2019/3/26 23:00:00
5 五班 2019/3/26 23:00:00

5.日期范围比较(格式化)

查询语句

--显示转换
SELECT *
  FROM T_CLASS T
 WHERE TO_CHAR(T.CREATE_TIME, 'YYYY-MM-DD') 
    BETWEEN '2019-03-01' AND '2019-03-26';

查询结果

CLASS_ID CLASS_NAME CREATE_TIME
1 一班 2019/3/26 23:00:00
2 二班 2019/3/26 23:00:00
3 三班 2019/3/26 23:00:00
4 四班 2019/3/26 23:00:00
5 五班 2019/3/26 23:00:00

四、分析

对 Oracle 日期字段的查询,建议通过 to_char 或者 to_date 这两个方法将相应的数据进行格式化。
对数据进行格式化的好处:

  1. 可以使查询语句一目了然。
  2. 避免因为隐式转换带来的业务逻辑上的风险。
  3. 避免了因为调试问题而导致的时间浪费(如果不是熟悉Oracle,边界问题也是比较难发现的)

如果文章有帮助到了你,欢迎点赞、转发。

如果文章有错误的地方,欢迎留言交流。

相关文章
|
8月前
|
SQL Oracle 关系型数据库
Oracle之日期计算相关函数
Oracle之日期计算相关函数
90 0
|
5月前
|
Oracle 关系型数据库 数据库
oracle日期加减的三种方式
【8月更文挑战第15天】在Oracle数据库中,可通过三种方式对日期进行加减操作:一是利用`ADD_MONTHS`函数增减月份,如`ADD_MONTHS(SYSDATE, 2)`表示两个月后;二是运用算术运算符直接加减天数,如`SYSDATE + 7`表示七天后;三是采用`INTERVAL`表达式增加或减少特定时间间隔,如`SYSDATE + INTERVAL '2' YEAR`表示两年后。这些方法可根据实际需求灵活选用。
2699 5
|
6月前
|
SQL Oracle 算法
|
8月前
|
Oracle 关系型数据库 数据管理
Oracle常用系统函数之日期和时间类函数:时空穿梭者的魔法棒
【4月更文挑战第19天】Oracle数据库中的日期和时间函数是强大的工具,如同时空穿梭者的魔法棒。`SYSDATE`和`CURRENT_TIMESTAMP`能显示当前时间,助你在数据中记录关键时刻;`ADD_MONTHS`函数让你轻易跃过月份,实现时间穿越;而`TO_DATE`和`TO_CHAR`则负责日期和时间的格式转换,如同时间的化妆师。除此之外,还有`EXTRACT`、`LAST_DAY`等更多魔法道具,帮助数据管理专家们自如操控时间,解决各种挑战。掌握这些函数,你也能在数据世界中自由穿梭!
|
SQL 存储 Oracle
Oracle数据库中日期的操作、主键自增与分页查询
Oracle数据库中日期的操作、主键自增与分页查询
123 0
|
8月前
|
SQL Oracle 关系型数据库
Oracle之有哪些日期计算函数?
Oracle之有哪些日期计算函数?
155 0
|
8月前
|
SQL Oracle 关系型数据库
oracle中日期循环
oracle中日期循环
397 0
|
SQL Oracle 关系型数据库
sql中datetime日期类型字段比较(mysql&oracle)
sql中datetime日期类型字段比较(mysql&oracle)
284 1
|
Oracle 关系型数据库 数据库
Oracle 数据库中常见的日期和时间函数
Oracle 数据库中常见的日期和时间函数
216 0
|
Oracle 关系型数据库
Oracle日期加减运算实战演练
Oracle日期加减运算实战演练
123 0

热门文章

最新文章

推荐镜像

更多