如何使用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方法進行字符串拼接,方便了年份替換,可以直接替換置對應的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,當然也可以使用DATE_SUB或者去YEAR和MONTH信息再進行拼接;
LAST_DAY(date_value):data_value所在月的最后一天。
一個月的解決了,那么多個月的無非就手寫幾個范圍就可以了(x
當然不能手寫這些范圍,一方面是很麻煩而且不好看,另一方面是會給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ù)的返回值如下,當 N < n 1 N < n1 N<n1返回0,當 n 1 ≤ N < n 2 n_1 \leq N < n_2 n1?≤N<n2?時返回1,當 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ù)記錄進當年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í)行完當前這步,可以獲取的結(jié)果如下:
3.統(tǒng)計每月數(shù)據(jù)
在ct這一列中,我們獲取的數(shù)據(jù)是有規(guī)律的,比如一個日期中會有兩個"-"、兩個日期之間以",“分隔。
這里我們選擇以”,"為標志,統(tǒng)計出有多少個分隔符,再+1就得到了數(shù)據(jù)的數(shù)量。
至于實現(xiàn)方式,可以使用如下方式:
即
LENGTH(tmp.ct) - LENGTH(REPLACE(tmp.ct, ',', '')) + 1
4.統(tǒng)計值與月份相對應
取得GROUP_CONCAT獲取的第一個日期即可代表這一整個數(shù)據(jù)所在的月份。
可以使用SUBSTRING_INDEX()函數(shù),它有三個參數(shù),第一個參數(shù)為待片取的字符串、第二個參數(shù)為分隔符、第三個參數(shù)為第幾個截取到第幾個分隔符。
如此一來:
SUBSTRING_INDEX(tmp.ct, ',', 1)
便可以取到該日期,再使用MONTH函數(shù)即可獲取對應的月份。
5.總體整合
我這里是使用了一次子查詢,子查詢獲取對應的分組及GROUP_CONCAT數(shù)據(jù),再交由外查詢進行處理。
結(jié)語
這里給出的方案僅僅是一種方案,也許存在著其他更快更好的解決方案但我沒有想到,在復雜問題面前一步一步獲取小數(shù)據(jù)是我習慣,這也就使得很可能出現(xiàn)多個嵌套著的子查詢。
到此這篇關(guān)于如何使用MySQL查詢一年中每月的記錄數(shù)的文章就介紹到這了,更多相關(guān)MySQL查詢每月記錄數(shù)內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
MySql數(shù)據(jù)庫基礎(chǔ)之分組查詢詳解
這篇文章主要介紹了mysql按照時間分組查詢的語句,非常實用,sql語句簡單易懂,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2022-09-09關(guān)于Mysql-connector-java驅(qū)動版本問題總結(jié)
這篇文章主要介紹了Mysql-connector-java驅(qū)動版本問題,本文給大家介紹的很詳細,通過原因說明問題小結(jié)個人建議給大家展示的很好,需要的朋友可以參考下2021-06-06mysql 忘記密碼的解決方法(linux和windows小結(jié))
下面是linux和windows下mysql丟失密碼的解決辦法2008-12-12rpm -ivh方式安裝mysql并修改數(shù)據(jù)存儲位置的實現(xiàn)
在Linux環(huán)境下進行MySQL的安裝可以使用不同的方式,但在本文中我們將關(guān)注一種特定的方式,即通過RPM包的方式進行安裝,本文主要介紹了rpm -ivh方式安裝mysql并修改數(shù)據(jù)存儲位置的實現(xiàn),感興趣的可以了解一下2023-09-09