背景:在開發過程中,我們經常需要根據時間作為判斷條件來查詢數據,例如:當月,當日,當前小時,幾天內......
1. 當月
我們只需要使用一個mysql的MONTH(date)
函數即可實現。(注意判斷年份)
MONTH(date);
-- 用法:MONTH函數返回一個整數,表示指定日期值的月份。
-- 舉例
SELECT MONTH('2020-11-11 00:00:00')
-- 返回值是11
2. 30天內
之所以把“30天內”放在當月的后面,是因為我經常會遇到這兩個需求相互轉換的情況,“30天內”也可以稱作“一個月內”。
這種情況我們需要使用DATEDIFF(expr1,expr2)
函數。
DATEDIFF(expr1,expr2)
-- 用法:參數為兩個日期,返回的是expr1-expr2的天數差
-- 舉例
SELECT DATEDIFF('2020-11-01 08:00:00','2020-11-11 00:00:00')
-- 返回值是-10
3. 當日
當日需要使用TO_DAYS(date)
函數。
TO_DAYS(date)
-- 用法:返回從0000年(公元1年)至當前日期的總天數。
-- 舉例
SELECT TO_DAYS('0000-01-01')
-- 返回值是1
SELECT TO_DAYS('0001-01-01')
-- 返回值是366
4. 當前小時
這種情況需要HOUR(date)
和CURDATE()
函數配合使用。
HOUR(date)
-- 用法:返回當前時間是今日的第幾個小時
-- 舉例
SELECT HOUR('2020-11-11 11:11:11')
-- 返回值是11
CURDATE()
-- 用法:返回今日的日期,不包括時分秒, yyyy-MM-dd
-- 使用舉例
select * from table where created_at > CURDATE() and HOUR(created_at) = HOUR(now())
5. x天內
可以使用DATE_SUB(date,INTERVAL expr unit)
函數來實現。
DATE_SUB(date,INTERVAL expr unit)
-- 用法:起始日期date 減去一個時間段后的日期
后面的單位unit有很多值可以選擇,如下表:
類型(unit值) | 含義 | expr表達式的形式 |
---|---|---|
YEAR | 年 | YY |
MONTH | 月 | MM |
DAY | 日 | DD |
HOUR | 時 | hh |
MINUTE | 分 | mm |
SECOND | 秒 | ss |
YEAR_MONTH | 年和月 | YY和MM之間用任意符號隔開 |
DAY_HOUR | 日和小時 | DD和hh之間用任意符號隔開 |
DAY_MINUTE | 日和分鍾 | DD和mm之間用任意符號隔開 |
DAY_SECOND | 日和秒鍾 | DD和ss之間用任意符號隔開 |
HOUR_MINUTE | 時和分 | hh和mm之間用任意符號隔開 |
HOUR_SECOND | 時和秒 | hh和ss之間用任意符號隔開 |
MINUTE_SECOND | 分和秒 | mm和ss之間用任意符號隔開 |
-- 舉例
-- 七天內的數據查詢
select * from table where created_at > DATE_SUB(CURDATE(),INTERVAL 7 DAY)
6. 多少天內數據統計
我們經常還會遇到這種需求,統計7天內每天數據的量。這種情況下,我們需要考慮沒有數據推送的情況,即為0也要得到。
我的思路如下:
SELECT
DATE_SUB( CURDATE(), INTERVAL m.s day ) AS orderDate
FROM
(
SELECT 0 as s UNION ALL
SELECT 1 UNION ALL
SELECT 2 UNION ALL
SELECT 3 UNION ALL
SELECT 4 UNION ALL
SELECT 5 UNION ALL
SELECT 6
) m
以上面的sql查詢結果作為臨時表,匹配數據表統計多少天內的數據數量
-- 舉例:查詢12個月內每個月數據的數量
SELECT COUNT(t.created_at),res.date FROM
(SELECT
DATE_FORMAT(DATE_SUB( CURDATE(), INTERVAL m.s MONTH ),'%Y-%m') AS date
FROM
(
SELECT 0 as s UNION ALL
SELECT 1 UNION ALL
SELECT 2 UNION ALL
SELECT 3 UNION ALL
SELECT 4 UNION ALL
SELECT 5 UNION ALL
SELECT 6 UNION ALL
SELECT 7 UNION ALL
SELECT 8 UNION ALL
SELECT 9 UNION ALL
SELECT 10 UNION ALL
SELECT 11
) m)res
left join
table t on res.date = DATE_FORMAT(t.created_at,'%Y-%m')
GROUP BY date