MySQL的STR_TO_DATE函数和DATE_FORMAT函数

场景:使用STR_TO_DATE函数和DATE_FORMAT函数

记录:NO.249

 

体验一下STR_TO_DATE函数和DATE_FORMAT函数

 

STR_TO_DATE函数将str转化为日期型的数据,format表示转化后的格式。

DATE_FORMAT函数是根据format指定的格式显示date值。

 

时间类型转换为字符串

SELECT DATE_FORMAT(SYSDATE(),'%Y-%c-%d %H:%i:%s');

执行结果: 2021-6-20 21:00:06

SELECT DATE_FORMAT(SYSDATE(),'%Y-%m-%d %H:%i:%s');

执行结果: 2021-06-20 21:01:17

SELECT DATE_FORMAT(SYSDATE(),'%Y-%m-%d');

执行结果: 2021-06-20

SELECT DATE_FORMAT(SYSDATE(),'%Y-%m');

执行结果: 2021-06

SELECT DATE_FORMAT(SYSDATE(),'%Y');

执行结果: 2021

 

字符串转换为时间类型

SELECT STR_TO_DATE('2021-06-21 15:16:29','%Y-%m-%d %H:%i:%s');

执行结果: 2021-06-21 15:16:29

SELECT STR_TO_DATE('2021-06-21 15:16:29','%Y-%m-%d');

执行结果: 2021-06-21

 

format格式

 

示例:

SELECT NOW();

执行结果:  2021-06-20 23:14:20

SELECT DATE_FORMAT(SYSDATE(),'%a');

执行结果:  Sun

SELECT DATE_FORMAT(SYSDATE(),'%b');

执行结果:  Jun

SELECT DATE_FORMAT(SYSDATE(),'%c');

执行结果:  6

SELECT DATE_FORMAT(SYSDATE(),'%D');

执行结果:  20th

SELECT DATE_FORMAT(SYSDATE(),'%d');

执行结果:  20

SELECT DATE_FORMAT(SYSDATE(),'%e');

执行结果:  20

SELECT DATE_FORMAT(SYSDATE(),'%f');

执行结果:  000000

SELECT DATE_FORMAT(SYSDATE(),'%H');

执行结果:  23

SELECT DATE_FORMAT(SYSDATE(),'%h');

执行结果:  11

SELECT DATE_FORMAT(SYSDATE(),'%i');

执行结果:  17

SELECT DATE_FORMAT(SYSDATE(),'%j');

执行结果:  171

SELECT DATE_FORMAT(SYSDATE(),'%k');

执行结果:  23

SELECT DATE_FORMAT(SYSDATE(),'%l');

执行结果:  11

SELECT DATE_FORMAT(SYSDATE(),'%M');

执行结果:  June

SELECT DATE_FORMAT(SYSDATE(),'%m');

执行结果:  06

SELECT DATE_FORMAT(SYSDATE(),'%p');

执行结果:  PM

SELECT DATE_FORMAT(SYSDATE(),'%r');

执行结果:  11:19:32 PM

SELECT DATE_FORMAT(SYSDATE(),'%S');

执行结果:  02

SELECT DATE_FORMAT(SYSDATE(),'%s');

执行结果:  16

SELECT DATE_FORMAT(SYSDATE(),'%T');

执行结果:  23:20:31

SELECT DATE_FORMAT(SYSDATE(),'%U');

执行结果:  25

SELECT DATE_FORMAT(SYSDATE(),'%u');

执行结果:  24

SELECT DATE_FORMAT(SYSDATE(),'%V');

执行结果:  25

SELECT DATE_FORMAT(SYSDATE(),'%v');

执行结果:  24

SELECT DATE_FORMAT(SYSDATE(),'%W');

执行结果:  Sunday

SELECT DATE_FORMAT(SYSDATE(),'%w');

执行结果:  0

SELECT DATE_FORMAT(SYSDATE(),'%X');

执行结果:  2021

SELECT DATE_FORMAT(SYSDATE(),'%x');

执行结果:  2021

SELECT DATE_FORMAT(SYSDATE(),'%Y');

执行结果:  2021

SELECT DATE_FORMAT(SYSDATE(),'%y');

执行结果:  21

SELECT DATE_FORMAT(SYSDATE(),'%%');

执行结果:  %

 

<think>好的,我现在需要帮助用户解决他们关于MySQLSTR_TO_DATE函数使用的问题,以及为什么他们的SQL查询没有返回数据。用户之前的问题是关于DATE_FORMAT的错误使用,现在他们转向了STR_TO_DATE,可能还存在类似的日期格式问题。首先,我需要回顾用户提供的当前查询:SELECT *FROM qqt_lecture_act_recordWHERE STR_TO_DATE(action_time, '%y%M%d') ='20250527'AND activity_code ='23' 用户的问题有两个部分:为什么查询不到数据,以及如何正确使用STR_TO_DATE函数。我需要先分析STR_TO_DATE的使用是否正确,以及日期格式是否匹配。根据之前的回答,用户之前使用DATE_FORMAT时,格式符错误导致匹配失败。现在换成了STR_TO_DATE,可能需要检查格式符是否正确。STR_TO_DATE的作用是将字符串转换为日期,但这里action_time字段可能已经是日期类型,所以使用STR_TO_DATE可能不必要,甚至导致错误。接下来,我需要确认STR_TO_DATE的正确用法。STR_TO_DATE的第一个参数应该是字符串,如果action_time已经是日期或时间类型,那么使用这个函数可能会将日期转换为字符串再转换回来,导致不可预期的结果。例如,如果action_time是datetime类型,直接使用STR_TO_DATE可能不会按预期工作,反而可能因为格式不匹配返回NULL,导致条件不成立。 然后,检查格式字符串'%y%M%d'。%y是两位数的年份,%M是月份的英文全名,%d是日期。例如,如果action_time是'2025-05-27',STR_TO_DATE会将其转换为字符串,但格式符%y会得到25,%M得到May,%d得到27,组合成25May27,而用户希望比较的是20250527,显然不匹配。因此,正确的格式符应该是'%Y%m%d',其中%Y是四位年份,%m是两位月份,%d是两位日期。 另外,用户的条件是STR_TO_DATE的结果等于'20250527',但STR_TO_DATE返回的是日期类型,而'20250527'是字符串,比较时MySQL会进行隐式转换,可能不如预期。正确的做法是将字符串也转换为日期,或者直接使用日期字面量。例如,使用STR_TO_DATE('20250527','%Y%m%d')来生成正确的日期值进行比较。此外,使用函数在WHERE子句中对列进行操作会导致索引失效,影响性能。建议改为使用范围查询,比如action_time >='2025-05-2700:00:00' ANDaction_time< '2025-05-2800:00:00',这样可以有效利用索引。另外需要考虑时区问题,如果action_time存储的是UTC时间,而查询使用的是本地时间(如UTC+8),可能需要使用时区转换函数,如CONVERT_TZ,来调整时间。总结问题可能的原因: 1. STR_TO_DATE的格式符错误,导致转换后的日期与目标字符串不匹配。2.对日期类型的列使用STR_TO_DATE函数,导致转换错误或返回NULL。 3.隐式类型转换导致比较失败。4.时区差异导致实际存储的日期与查询条件不符。 解决方案: 1.修正格式符为'%Y%m%d'。2.避免在列上使用函数,改用范围查询。 3.确保比较的双方都是正确的日期类型。 4.检查时区设置,必要时进行转换。 现在需要将这些分析整理成结构化的回答,并给出示例代码注意事项。同时,生成相关问题,帮助用户深入理解。</think>### 为什么SQL查询无法返回数据? #### 一、核心问题分析 1. **`STR_TO_DATE`函数误用** - **函数作用混淆**:`STR_TO_DATE()`用于**将字符串转换为日期**,但`action_time`字段本身已是日期/时间类型时,此操作会导致MySQL先将日期隐式转为字符串,再尝试转换回日期,极易出错[^1]。 - **错误格式符**: ```sql STR_TO_DATE(action_time, '%y%M%d') -- 错误格式符组合 ``` - `%y`:两位年份(如2025转为`25`) - `%M`:月份英文名(如五月转为`May`) - `%d`:两位日期(如`27`) 实际转换结果为类似`25May27`的字符串,无法匹配`20250527`[^1]。 2. **隐式类型转换问题** - `STR_TO_DATE()`返回`DATE`类型,而条件值`'20250527'`是字符串,MySQL会尝试将日期再次转为字符串进行比较,导致格式不匹配(如日期转为`2025-05-27`与`20250527`不相等)[^1]。 --- ### 二、`STR_TO_DATE()`正确用法及修复方案 #### 1. 正确语法与格式符 ```sql -- 将字符串转为日期 SELECT STR_TO_DATE('20250527', '%Y%m%d'); -- 输出: 2025-05-27 ``` - **关键格式符**: - `%Y`:四位年份(如`2025`) - `%m`:两位月份(如`05`) - `%d`:两位日期(如`27`) #### 2. 修复后的查询语句 ```sql SELECT * FROM qqt_lecture_act_record WHERE DATE(action_time) = STR_TO_DATE('20250527', '%Y%m%d') -- 显式类型匹配 AND activity_code = '23'; ``` - **优化建议**(避免索引失效): ```sql WHERE action_time >= '2025-05-27 00:00:00' AND action_time < '2025-05-28 00:00:00' ``` --- ### 三、常见错误场景 #### 1. 时区不一致 若`action_time`存储为UTC时间,而应用时区为东八区(UTC+8),需转换时区: ```sql WHERE DATE(CONVERT_TZ(action_time, '+00:00', '+08:00')) = '2025-05-27' ``` #### 2. 日期范围边界 避免使用`BETWEEN`(包含右边界时间点): $$ \text{正确范围} = [t_{\text{start}}, t_{\text{end}}) $$ --- ### 四、公式验证 假设表中存在日期`$t$`,验证逻辑为: $$ \text{查询条件成立} \iff \text{STR\_TO\_DATE}(t, \text{'%Y%m%d'}) = \text{STR\_TO\_DATE}(\text{'20250527'}, \text{'%Y%m%d'}) $$ ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值