在數(shù)據(jù)庫搬磚的過程中,免不了要跟日期打交道,比如按日期匯總一些指標、統(tǒng)計某段時間內(nèi)的總量等。
如果是固定的日期還好,只需直接指定即可,但很多時候都是需要根據(jù)當前日期自適應(yīng)變化的。比如:提取上周一到上周日的數(shù)據(jù)、提取上個月的數(shù)據(jù)、提取前N個月的數(shù)據(jù)。。。
這些要求都有一個共同點,就是要視當前日期而定!那么,我們就需要首先獲取當前日期的一些信息,如當前日期是本周第幾天、本月第幾天等,然后,才能做下一步處理。
一、在提取所需日期區(qū)間之前,我們先介紹幾個常用的函數(shù)
| -- 先運行這一句SET @t = '2018-07-26 11:41:29';-- 再運行這一句SELECT DATE(@t)當前日期,YEAR(@t) 年份, MONTH(@t)月份,WEEK(@t)本年第幾周, DAY(@t) 本月第幾天, HOUR(@t)小時, MINUTE(@t)分鐘, SECOND(@t)秒 |
運行出來,結(jié)果是這樣的:

二、下面介紹幾個常用的日期加減運算函數(shù)
1、ADDDATE(expr, days) / SUBDATE(expr, days):
ADDDAT函數(shù)有兩個形參,第一個是基準日期,也即需要運算的日期;第二個是一個間隔表達式,形如:INTERVAL 1 HOUR,其中INTERVAL 是間隔的意思,中間的數(shù)字1可以換成任意整數(shù),第三個hour可以替換成day/month/year等時間單位。
SUBDATE同理,只不過變成了減法運算。
完整的用法如下:
| SELECT ADDDATE('2018-07-26 11:41:29',INTERVAL 1 HOUR);SELECT SUBDATE('2018-07-26 11:41:29',INTERVAL 1 HOUR); |
2、DATE_ADD() / DATE_SUB():
用法同ADDDATE(expr, days) / SUBDATE(expr, days)。
三、日期區(qū)間截取
接下來,利用上面介紹的日期函數(shù)的用法,就可以進行日期區(qū)間的截取了。
1、上周
| -- 提取上周的日期區(qū)間SELECT CURDATE() NOW,ADDDATE(ADDDATE(DATE_SUB(CURDATE(),INTERVAL WEEKDAY(CURDATE()) + 1 DAY),-6),0) startdate,ADDDATE(DATE_SUB(CURDATE(),INTERVAL WEEKDAY(CURDATE()) + 1 DAY),0) duetodate |
2、上月
| -- 上月SELECT CURDATE() NOW, DATE_ADD(DATE_ADD(CURDATE(),INTERVAL -DAY(CURDATE())+1 DAY), INTERVAL -1 MONTH) startdate,DATE_ADD(DATE_ADD(CURDATE(),INTERVAL -DAY(CURDATE()) DAY), INTERVAL 0 MONTH) duetodate |
3、前四個月
| -- 前四個月SELECT CURDATE() NOW, ADDDATE(ADDDATE(CURDATE(),INTERVAL -DAY(CURDATE())+1 DAY), INTERVAL -4 MONTH) startdate,DATE_ADD(DATE_ADD(CURDATE(),INTERVAL -DAY(CURDATE()) DAY), INTERVAL 0 MONTH) duetodate |
4、上周四到本周三
有時候不是按照自然周進行統(tǒng)計的,需要自定義一周的起止日期,可以按照下面的方式:
| SELECT ADDDATE(DATE_SUB(CURDATE(),INTERVAL (IF(WEEKDAY(CURDATE())IN(3,4,5,6),WEEKDAY(CURDATE()),WEEKDAY(CURDATE())+7 )) + 1 DAY),-3) startdate, ADDDATE(DATE_SUB(CURDATE(),INTERVAL (IF(WEEKDAY(CURDATE())IN(3,4,5,6),WEEKDAY(CURDATE()),WEEKDAY(CURDATE())+7 )) + 1 DAY),3)duetodate |