如何使用MySQL查詢一年中每月的記錄數(shù)
以下演示將在下表數(shù)據(jù)中進行:

其中:id為主鍵用于表的連接;value1為需要統(tǒng)計的主體,如用戶等;date為記錄日期。
先說結(jié)論
SELECT
tmp.value1 AS `value1`,
MONTH(SUBSTRING_INDEX(tmp.ct, ',', 1)) AS `month`,
LENGTH(tmp.ct) - LENGTH(
REPLACE
(tmp.ct, ',', '')
) + 1 AS `cnt`
FROM
(
SELECT
id,
value1,
GROUP_CONCAT(date_value) AS ct
FROM
test_year_record
GROUP BY
value1,
INTERVAL(
date_value,
DATE(CONCAT('2022', '-01-01')),
DATE(CONCAT('2022', '-02-01')),
DATE(CONCAT('2022', '-03-01')),
DATE(CONCAT('2022', '-04-01')),
DATE(CONCAT('2022', '-05-01')),
DATE(CONCAT('2022', '-06-01')),
DATE(CONCAT('2022', '-07-01')),
DATE(CONCAT('2022', '-08-01')),
DATE(CONCAT('2022', '-09-01')),
DATE(CONCAT('2022', '-10-01')),
DATE(CONCAT('2022', '-11-01')),
DATE(CONCAT('2022', '-12-01')),
DATE(CONCAT('2023', '-01-01'))
)
) AS tmp
JOIN test_year_record AS ot
ON
ot.id = tmp.id
WHERE
ot.value1 = 1 AND YEAR(SUBSTRING_INDEX(tmp.ct, ',', 1)) = '2022'注:以’2022’為例,上面結(jié)論中使用了CONCAT方法進行字符串拼接,方便了年份替換,可以直接替換置對應(yīng)的ORM的參數(shù)等。
查詢結(jié)果

思路及SQL解釋
這個問題可以劃分為如下幾個子問題,我們可以挨個分析解決:
1. 如何以月份劃分
對于一個月份的數(shù)據(jù)可以如下判斷:
date_value >= DATE_ADD(date_value, INTERVAL - DAY(date_value) + 1 DAY) AND data_value <= LAST_DAY(data_value)
解釋一下:
DATE_ADD(date_value, INTERVAL - DAY(date_value) + 1 DAY):data_value所在月的第一天,原理為在data_value的基礎(chǔ)上加上-DAY(data_value)天數(shù)再+1,當(dāng)然也可以使用DATE_SUB或者去YEAR和MONTH信息再進行拼接;
LAST_DAY(date_value):data_value所在月的最后一天。
一個月的解決了,那么多個月的無非就手寫幾個范圍就可以了(x
當(dāng)然不能手寫這些范圍,一方面是很麻煩而且不好看,另一方面是會給mysql帶來過多的計算量。
那么如何給12月進行劃分呢:
INTERVAL() 函數(shù)可以解決我們的問題:
INTERVAL( N , n 1 , n 2 , ? , n 3 N,n_1,n_2,\cdots,n_3 N,n1?,n2?,?,n3?),其中 N N N為帶判斷是數(shù)據(jù),后面的 n 1 ∼ n n n_1 \sim n_n n1?∼nn?分別為各個間斷點,這個函數(shù)的返回值如下,當(dāng) N < n 1 N < n1 N<n1返回0,當(dāng) n 1 ≤ N < n 2 n_1 \leq N < n_2 n1?≤N<n2?時返回1,當(dāng) n 2 ≤ N < n 3 n_2 \leq N < n_3 n2?≤N<n3?時返回2,…,以此類推。
據(jù)此,我們可以給一年做一個分段:
INTERVAL(
date_value,
DATE(CONCAT('2022', '-01-01')), # 一月
DATE(CONCAT('2022', '-02-01')), # 二月
DATE(CONCAT('2022', '-03-01')), # 三月
DATE(CONCAT('2022', '-04-01')), # 四月
DATE(CONCAT('2022', '-05-01')), # 五月
DATE(CONCAT('2022', '-06-01')), # 六月
DATE(CONCAT('2022', '-07-01')), # 七月
DATE(CONCAT('2022', '-08-01')), # 八月
DATE(CONCAT('2022', '-09-01')), # 九月
DATE(CONCAT('2022', '-10-01')), # 十月
DATE(CONCAT('2022', '-11-01')), # 十一月
DATE(CONCAT('2022', '-12-01')), # 十二月
DATE(CONCAT('2023', '-01-01')) # 次年一月,防止次年的數(shù)據(jù)記錄進當(dāng)年12月中
)
注: 這里其實還有個問題,就是結(jié)果會返回去年的數(shù)據(jù)(0),可以像我一樣在外查詢里面進行一個年份判斷,也可以交給java等檢測。
2.獲取每月數(shù)據(jù)
可以使用GROUP BY子句,以INTERVAL的值進行分組(為了保證屬于同一個value1的數(shù)據(jù),還需要以value1進行分組)。
注:GROUP BY 子句中含有多個參數(shù)時,將會是多條這些數(shù)據(jù)都一樣的記錄分為一組。
僅僅是做了分組是不夠的,我們還需要GROUP_CONCAT()函數(shù)來獲取一個分組中的數(shù)據(jù)集。
執(zhí)行完當(dāng)前這步,可以獲取的結(jié)果如下:

3.統(tǒng)計每月數(shù)據(jù)
在ct這一列中,我們獲取的數(shù)據(jù)是有規(guī)律的,比如一個日期中會有兩個"-"、兩個日期之間以",“分隔。
這里我們選擇以”,"為標(biāo)志,統(tǒng)計出有多少個分隔符,再+1就得到了數(shù)據(jù)的數(shù)量。
至于實現(xiàn)方式,可以使用如下方式:

即
LENGTH(tmp.ct) - LENGTH(REPLACE(tmp.ct, ',', '')) + 1
4.統(tǒng)計值與月份相對應(yīng)
取得GROUP_CONCAT獲取的第一個日期即可代表這一整個數(shù)據(jù)所在的月份。
可以使用SUBSTRING_INDEX()函數(shù),它有三個參數(shù),第一個參數(shù)為待片取的字符串、第二個參數(shù)為分隔符、第三個參數(shù)為第幾個截取到第幾個分隔符。
如此一來:
SUBSTRING_INDEX(tmp.ct, ',', 1)
便可以取到該日期,再使用MONTH函數(shù)即可獲取對應(yīng)的月份。
5.總體整合
我這里是使用了一次子查詢,子查詢獲取對應(yīng)的分組及GROUP_CONCAT數(shù)據(jù),再交由外查詢進行處理。
結(jié)語
這里給出的方案僅僅是一種方案,也許存在著其他更快更好的解決方案但我沒有想到,在復(fù)雜問題面前一步一步獲取小數(shù)據(jù)是我習(xí)慣,這也就使得很可能出現(xiàn)多個嵌套著的子查詢。
到此這篇關(guān)于如何使用MySQL查詢一年中每月的記錄數(shù)的文章就介紹到這了,更多相關(guān)MySQL查詢每月記錄數(shù)內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
MySQL中Order By多字段排序規(guī)則代碼示例
這篇文章主要介紹了MySQL中Order By多字段排序規(guī)則代碼示例,小編覺得挺不錯的,這里給大家分享下,需要的朋友可以參考。2017-10-10
mysql視圖之創(chuàng)建視圖(CREATE VIEW)和使用限制實例詳解
這篇文章主要介紹了mysql視圖之創(chuàng)建視圖(CREATE VIEW)和使用限制,結(jié)合實例形式詳細分析了mysql視圖創(chuàng)建于使用相關(guān)原理與操作注意事項,需要的朋友可以參考下2019-12-12
mysql8.0.11 winx64安裝配置方法圖文教程(win10)
這篇文章主要為大家詳細介紹了win10下mysql8.0.11 winx64安裝配置方法圖文教程,具有一定的參考價值,感興趣的小伙伴們可以參考一下2018-05-05

