MySQL 日期函數、時間函數在實際場景中的應用

2023-01-24 06:00:42

整理日常業務中用到日期函數的一些場景,並對日期函數按照使用型別做了分類,範例也儘可能符合日常需求。為了方便查閱,可以先看目錄,再根據需要看具體方法和範例。
首先明確日期和時間型別有哪些,也就是日期函數括號中的變數型別。

一、日期和時間型別

表示時間值的日期和時間型別為DATETIME、DATE、TIMESTAMP、TIME和YEAR。

型別 格式 描述
DATE YYYY-MM-DD 日期值
TIME HH:MM:SS 時間值或持續時間
YEAR YYYY 年份值
DATETIME YYYY-MM-DD hh:mm:ss 混合日期和時間值
TIMESTAMP YYYY-MM-DD hh:mm:ss 混合日期和時間值,時間戳

二、具體方法

(1)獲取年、月、日、周等

函數 描述
date(datetime) 返回datetime中的日期date部分
year(date) 返回年份
quarter(date) 返回日期date是第幾季節,返回 1 到 4
month(date) 返回日期date中的月份值,1 到 12
week(date) 計算日期 date 是本年的第幾個星期,範圍是 0 到 53
day(date) 返回日期值 date 的日期部分
hour(time) 返回 date 中的小時值
minute(time) 返回 date 中的分鐘值
second(time) 返回 t 中的秒鐘值
microsecond(time) 返回日期引數所對應的微秒數
extract(type from date) 從日期 date 中獲取指定的值,type 指定返回的值
weekday(date) 日期 date 是星期幾,0 表示星期一,1 表示星期二...

本文中type的取值:

year(年)、quarter(季)、month(月)、week(周)、day(天)、hour(小時)、minute(分鐘)、second(秒)、microsecond(毫秒)。

(2)獲取年月、年月日等

函數 描述
year、month、day 使用and連線單個返回值
date_format(event_time,’unit‘) 按照unit取值
substring(date, start, length) 從字串 str 的 start 位置開始擷取長度為 length 的子字串
like 模糊查詢

本文中unit的取值:

%M 月名字(January……December) 
%W 星期名字(Sunday……Saturday) 
%D 有英語字首的月份的日期(1st, 2nd, 3rd, 等等。) 
%Y 年, 數位, 4 位 
%y 年, 數位, 2 位 
%a 縮寫的星期名字(Sun……Sat) 
%d 月份中的天數, 數位(00……31) 
%e 月份中的天數, 數位(0……31) 
%m 月, 數位(01……12) 
%c 月, 數位(1……12) 
%b 縮寫的月份名字(Jan……Dec) 
%j 一年中的天數(001……366) 
%H 小時(00……23) 
%k 小時(0……23) 
%h 小時(01……12) 
%I 小時(01……12) 
%l 小時(1……12) 
%i 分鐘, 數位(00……59) 
%r 時間,12 小時(hh:mm:ss [AP]M) 
%T 時間,24 小時(hh:mm:ss) 
%S 秒(00……59) 
%s 秒(00……59) 
%p AM或PM 
%w 一個星期中的天數(0=Sunday ……6=Saturday ) 
%U 星期(0……52), 這裡星期天是星期的第一天 
%u 星期(0……52), 這裡星期一是星期的第一天 
%% 一個文字「%」。 

例如:date_format(%Y-%m-%d %H:%i:%s)得到時間戳

(3)兩個日期時間的整數差

函數 描述
datediff(date_1,date_2) 返回日期 date_1 - date_2 之間相隔的天數
timestampdiff(type,datetime_1,datetime_2) 返回 datetime_2 − datetime_1 的時間差
timediff(time_1, time_2)計算時間差值 返回time_1 - time_2 的時間差值

引數說明:

type:引數確定(end-begin)結果的單位,表示為整數。可以填year(年)、quarter(季)、month(月)、week(周)、day(天)、hour(小時)、minute(分鐘)、second(秒)、microsecond(毫秒)。

返回datetime_2-datetime_1的結果,其中datetime_1和datetime_2是date或者datetime表示式,也可以是混合型別;

當datetime_1>datetime_2時,返回的結果為負;

當datetime_1<datetime_2時,返回的結果為正。

  • 什麼時候使用datediff?什麼時候使用timestampdiff?什麼時候使用timediff?
三個函數的區別主要在於物件的不同一個是日期,一個是時間,另外一個既可以是日期格式也可以是時間格式。
timestampdiff更加靈活,既可以對日期求整數差也可以對時間求整數差。而datediff只能求日期的天數差。

(4)日期減去/加上天數得到日期 date_sub、date_add

函數 描述
date_add(date,interval expr type) 返回起始日期 date 加上一個時間段後的日期
date_sub(date,interval expr type) 返回函數從日期減去指定的時間間隔
addtime(time,n) 返回n 是一個時間表示式,時間 t 加上時間表示式 n
subtime(time,n) 時間 t 減去 n 秒的時間

(5)日期date是周幾

函數 描述
weekday(date) 日期 date 是星期幾,0 表示星期一,1 表示星期二...
date_format(date,'%W') 日期 date 是星期幾,Monday週一,Tuesday週二...
week(date) 計算日期 date 是本年的第幾個星期,範圍是 0 到 53

(6)某個月的最後一天

函數 描述
last_day(date) 返回給定日期的那一月份的最後一天

(7)返回當前日期/時間

函數 描述
curdate()、current_date() 返回當前日期
curtime()、current_time() 返回當前時間
now() 返回當前日期和時間

三、範例

分為兩部分,簡單和複雜範例,簡單範例是為了便於理解函數的使用,複雜範例是結合業務場景給出了常見的問題。

1、簡單範例

  • 獲取年、月、日、周等
# 返回'2023-01-09 22:32:17'的 年月日等 部分
select
	date('2023-01-09 22:32:17');			  -- 2023-01-09
select
	year('2023-01-09 22:32:17');			  -- 2023
select
	quarter('2023-01-09 22:32:17');			  -- 1
select
	month('2023-01-09 22:32:17');			  -- 1
select
	week('2023-01-09 22:32:17');			  -- 2(本年第二週)
select
	day('2023-01-09 22:32:17');			      -- 9
select
	hour('2023-01-09 22:32:17');			  -- 22
select
	minute('2023-01-09 22:32:17');			  -- 32
select
	second('2023-01-09 22:32:17');			  -- 17
select
	extract(day from '2023-01-09 22:32:17');  -- 9
select
	weekday('2023-01-09 22:32:17');			-- 0(週一)
  • 獲取年月、年月日等
# 返回'2023-01-09 22:32:17'的 年月、年月日等部分
select 
	date_format('2023-01-09 22:32:17','%Y%m');						  -- 202301
select 
	date_format('2023-01-09 22:32:17','%y%m');						  -- 2301
select 
	date_format('2023-01-09 22:32:17','%Y-%m-%d');					  -- 2023-01-09
select 
	date_format('2023-01-09 22:32:17','%H:%i');					  	  -- 22:32
select 
	date_format('2023-01-09 22:32:17','%s');					  	  -- 17
select 
	substring('2023-01-09 22:32:17',1,7);						  	  -- 2023-01
select 
	substring('2023-01-09 22:32:17',1,10);						  	  -- 2023-01-09
select 
	substring('2023-01-09 22:32:17',12,5);						  	  -- 22:32
select 
	substring('2023-01-09 22:32:17',18,2);						  	  -- 17
  • 兩個日期時間的整數差/和
# 返回'2023-01-09 22:32:17'和 '2023-03-15 04:18:56'的 整數差或者和
select 
	datediff('2023-01-09 22:32:17','2023-03-15 04:18:56');			  	  -- -65
select 
	datediff('2023-03-15 04:18:56','2023-01-09 22:32:17');			  	  -- 65
	
select 
	timediff('2023-01-09 22:32:17','2023-03-15 04:18:56');			  	  -- -838:59:59
select 
	timediff('2023-03-15 04:18:56','2023-01-09 22:32:17');			  	  -- 838:59:59
	
select 
	timestampdiff(year,'2023-01-09 22:32:17','2023-03-15 04:18:56');	  -- 0
select 
	timestampdiff(year,'2023-03-15 04:18:56','2023-01-09 22:32:17');	  -- 0
	
select 
	timestampdiff(quarter,'2023-01-09 22:32:17','2023-03-15 04:18:56');	  -- 0
select 
	timestampdiff(quarter,'2023-03-15 04:18:56','2023-01-09 22:32:17');	  -- 0
	
select 
	timestampdiff(month,'2023-01-09 22:32:17','2023-03-15 04:18:56');	  -- 2
select 
	timestampdiff(month,'2023-03-15 04:18:56','2023-01-09 22:32:17');	  -- -2
	
select 
	timestampdiff(week,'2023-01-09 22:32:17','2023-03-15 04:18:56');	  -- 9
select 
	timestampdiff(week,'2023-03-15 04:18:56','2023-01-09 22:32:17');	  -- -9
	
select 
	timestampdiff(day,'2023-01-09 22:32:17','2023-03-15 04:18:56');	  	  -- 64
select 
	timestampdiff(day,'2023-03-15 04:18:56','2023-01-09 22:32:17');	  	  -- -64
	
select 
	timestampdiff(hour,'2023-01-09 22:32:17','2023-03-15 04:18:56');	  -- 1541
select 
	timestampdiff(hour,'2023-03-15 04:18:56','2023-01-09 22:32:17');	  -- -1541
	
select 
	timestampdiff(minute,'2023-01-09 22:32:17','2023-03-15 04:18:56');	  -- 92506
select 
	timestampdiff(minute,'2023-03-15 04:18:56','2023-01-09 22:32:17');	  -- -92506
	
select 
	timestampdiff(second,'2023-01-09 22:32:17','2023-03-15 04:18:56');	  -- 5550399
select 
	timestampdiff(second,'2023-03-15 04:18:56','2023-01-09 22:32:17');	  -- -5550399
  • 日期減去/加上天數得到日期 date_sub、date_add
# 返回日期'2023-01-09 22:32:17'加上或者減去 某個時間段 的日期
select 
	date_add('2023-01-09 22:32:17',interval 2 year);			  -- 2025-01-09 22:32:17
select 
	date_add('2023-01-09 22:32:17',interval 2 quarter);			  -- 2023-07-09 22:32:17
select 
	date_add('2023-01-09 22:32:17',interval 2 month);		  	  -- 2023-03-09 22:32:17
select 
	date_add('2023-01-09 22:32:17',interval 2 week);		  	  -- 2023-01-23 22:32:17
select 
	date_add('2023-01-09 22:32:17',interval 2 day);			  	  -- 2023-01-11 22:32:17
select 
	date_add('2023-01-09 22:32:17',interval 2 hour);			  -- 2023-01-10 00:32:17
select 
	date_add('2023-01-09 22:32:17',interval 2 second);			  -- 2023-01-09 22:32:19
select 
	date_add('2023-01-09 22:32:17',interval 2 minute);			  -- 2023-01-09 22:34:17

select 
	date_sub('2023-01-09 22:32:17',interval 2 year);			  -- 2025-01-09 22:32:17
select 
	date_sub('2023-01-09 22:32:17',interval 2 quarter);			  -- 2023-07-09 22:32:17
select 
	date_sub('2023-01-09 22:32:17',interval 2 month);		  	  -- 2023-03-09 22:32:17
select 
	date_sub('2023-01-09 22:32:17',interval 2 week);		  	  -- 2023-01-23 22:32:17
select 
	date_sub('2023-01-09 22:32:17',interval 2 day);			  	  -- 2023-01-11 22:32:17
select 
	date_sub('2023-01-09 22:32:17',interval 2 hour);			  -- 2023-01-10 00:32:17
select 
	date_sub('2023-01-09 22:32:17',interval 2 second);			  -- 2023-01-09 22:32:19
select 
	date_sub('2023-01-09 22:32:17',interval 2 minute);			  -- 2023-01-09 22:34:17
# 時間格式預設秒
select 
	addtime('2023-01-09 22:32:17','22:12:00');			  		  -- 2023-01-10 20:44:17
select 
	addtime('2023-01-09 22:32:17','22:12');			  			  -- 2023-01-10 20:44:17
select 
	addtime('2023-01-09 22:32:17','22');			  			  -- 2023-01-09 22:32:39
	
select 
	subtime('2023-01-09 22:32:17','22:12:00');			  		  -- 2023-01-09 00:20:17
select 
	subtime('2023-01-09 22:32:17','22:12');			  			  -- 2023-01-09 00:20:17
select 
	subtime('2023-01-09 22:32:17','22');			  			  -- 2023-01-09 22:31:55
  • 日期date是周幾
# 返回日期'2023-01-09 22:32:17' 是周幾
select 
	weekday('2023-01-09 22:32:17');			  				  -- 0(週一)
select 
	week('2023-01-09 22:32:17');			  				  -- 2(本年第二週)
select 
	date_format('2023-01-09 22:32:17','%W');			  	  -- Monday(週一的英文)
  • 某個月的最後一天
# 返回日期'2023-01-09 22:32:17' 該月最後一天
select 
	last_day('2023-01-09 22:32:17');			  				  -- 2023-01-31
  • 返回當前日期/時間
# 返回當前日期或者時間
select 
	curdate(),			  				  	  -- 2023-01-23
	current_date(),			  				  -- 2023-01-23
	curtime(),			  				  	  -- 23:32:37
	current_time(),			  				  -- 23:32:37
	now();  								  -- 2023-01-23 23:32:37

2、複雜範例

students 表 (id表示主鍵,student_id是學號,score是平均成績,date是作答日期)

id student_id score answer_date
1 1123 98 2023-01-08
2 2341 74 2022-02-12
3 5623 66 2022-12-23
4 4386 80 2023-01-12

(1)統計每個月12號作答的次數——month、day

#使用month、day
select 
	month(answer_date) as month,
	count(answer_date) as count
from students_score
where day(answer_date) = 12
group by month;
  • 結果
month  count
1		1
2		1

(2)統計2023年1月作答的次數——year、month、day、date_format、substring

#1、使用year、month、day
select 
	count(answer_date) 
from students_score
where month(answer_date) = 1 and year(answer_date) = 2023;

#2、使用date_format
select 
	count(answer_date) 
from students_score
where date_format(answer_date,'%Y%m') = '202301';

#3、使用substring
select 
	count(answer_date) 
from students_score
where substring(answer_date,1,7) = '2023-01';

#4、使用like
select 
	count(answer_date) 
from students_score
where answer_date like '2023-01%';

#結果:2

(3)計算2022-02-12是2022年的第幾周——week(date)

select week('2022-02-12');   #6(表示第六週)

(4)計算2023-01-05是周幾——date_format(date,'%W')

select date_format('2023-01-05','%W');   #Thursday(表示週四)

(5)輸出students_score表中2023年在週一到週五作答的學生學號——weekday(date)

#1、使用weekday
select 
	student_id,
	weekday(answer_date) as week
from students_score
where weekday(answer_date) < 5;

#2、使用date_format(date,'%W')
select 
	student_id,
	date_format(answer_date,'%W') as week
from students_score
where date_format(answer_date,'%W') in ('Monday','Tuesday','Wednesday','Thursday','Friday');

注意兩種方法的區別

  • 結果
student_id  week
5623	4
4386	3

5623	Friday
4386	Thursday

(6)計算每月平均作答次數——last_day(date)

select 
	date_format(answer_date,'%Y%m') as date_,
	count(answer_date)/day(last_day(answer_date)) as avg_num
from students_score
group by date_;
  • 結果
date_	avg_num
202202	0.0357
202212	0.0323
202301	0.0645

(7)近7天和近30天分別作答題目數量

如何計算近7天和近30天的日期:日期減去天數得到日期;另一種是日期減去日期得到天數。

  • [使用]:date_sub(date,interval expr type)、datediff(date,dt_max) 、timestampdiff(day,date,dt_max)
#先求出今日是哪一天(最大日期),往前七天是哪一天,往前30天是哪一天
with 
	main as(
    	select
			max(date(answer_date)) as now_date,
			date_sub(max(date(answer_date)),interval 6 day) as 7_date,
			date_sub(max(date(answer_date)),interval 29 day) as 30_date
		from students_score
    )

select 
	sum(if(datediff(answer_date,7_date)) >= 0,1,0) as ans_7_num,
	sum(if(datediff(answer_date,30_date)) >= 0,1,0) as ans_30_num
from students_score, main
  • 結果
ans_7_num	ans_30_num
2				3

建立 students_score 表的程式碼

-- ----------------------------
-- Table structure for students_score
-- ----------------------------
DROP TABLE IF EXISTS `students_score`;
CREATE TABLE `students_score`  (
  `id` int(11) NOT NULL COMMENT '主鍵',
  `student_id` int(11) NOT NULL COMMENT '學號',
  `score` int(11) NULL DEFAULT NULL COMMENT '成績',
  `date` date NOT NULL COMMENT '作答日期',
  PRIMARY KEY (`id`) USING BTREE
) ENGINE = InnoDB CHARACTER SET = utf8 COLLATE = utf8_general_ci ROW_FORMAT = Dynamic;

-- ----------------------------
-- Records of students_score
-- ----------------------------
INSERT INTO `students_score` VALUES (1, 1123, 98, '2023-01-08');
INSERT INTO `students_score` VALUES (2, 2341, 74, '2023-01-12');
INSERT INTO `students_score` VALUES (3, 5623, 66, '2022-12-23');
INSERT INTO `students_score` VALUES (4, 4386, 80, '2023-01-12');