柚子快報(bào)邀請(qǐng)碼778899分享:【MySQL 的 ONLY
柚子快報(bào)邀請(qǐng)碼778899分享:【MySQL 的 ONLY
引言: 作為一個(gè)菜鳥(niǎo),當(dāng)寫(xiě)sql中涉及到group by這樣簡(jiǎn)單的語(yǔ)句時(shí),也會(huì)出現(xiàn)問(wèn)題,我在??途W(wǎng)上做sql題時(shí),總報(bào)這個(gè)錯(cuò): ONLY_FULL_GROUP_BY 到底是什么東西呢?
今天寫(xiě)篇文章解釋一下。
一、GROUP BY使用時(shí)的關(guān)鍵要點(diǎn)
1. 理解 GROUP BY 的作用
GROUP BY 用于將數(shù)據(jù)集分割成多個(gè)組,每個(gè)組由一組具有相同屬性的行組成。這使得聚合函數(shù)可以應(yīng)用于每一組,而不是整個(gè)數(shù)據(jù)集。例如,使用 GROUP BY 可以按產(chǎn)品類(lèi)別統(tǒng)計(jì)銷(xiāo)售總額。
2. 遵守 ONLY_FULL_GROUP_BY 規(guī)則
在 MySQL 中,如果啟用了 ONLY_FULL_GROUP_BY 模式,那么 SELECT 子句中除聚合函數(shù)之外的所有列都必須在 GROUP BY 子句中出現(xiàn)。這是為了避免不確定性和潛在的數(shù)據(jù)歧義。
3. 使用 HAVING 進(jìn)行條件過(guò)濾
HAVING 子句用于對(duì)分組后的結(jié)果進(jìn)行過(guò)濾,類(lèi)似于 WHERE 子句,但 HAVING 適用于聚合結(jié)果。例如,你可以使用 HAVING COUNT(*) > 1 來(lái)找出至少出現(xiàn)兩次的組。
4. 正確排序結(jié)果
雖然 GROUP BY 自身不會(huì)自動(dòng)排序結(jié)果,但你通常會(huì)希望在結(jié)果集中應(yīng)用 ORDER BY 來(lái)排序分組。例如,你可以按銷(xiāo)售額降序排序產(chǎn)品類(lèi)別。
5. 注意空值和 NULL 值
在 GROUP BY 中,NULL 值會(huì)被視為相同的值,這意味著所有包含 NULL 的行會(huì)被歸入同一組。如果需要區(qū)分 NULL 和非 NULL 值,可以使用 COALESCE() 或者條件表達(dá)式。
二、 ONLY_FULL_GROUP_BY 規(guī)則
1. 什么是 ONLY_FULL_GROUP_BY?
ONLY_FULL_GROUP_BY 是 MySQL 中的一個(gè) SQL 模式,它要求在任何包含聚合函數(shù)的查詢(xún)中,所有在 SELECT 子句中出現(xiàn)的非聚合列也必須在 GROUP BY 子句中出現(xiàn)。換句話(huà)說(shuō),如果一個(gè)查詢(xún)使用了聚合函數(shù),那么除了聚合函數(shù)包裹的列以外,所有在 SELECT 子句中出現(xiàn)的列都必須被 GROUP BY 子句引用。
這個(gè)規(guī)則確保了查詢(xún)結(jié)果的確定性和一致性,避免了由于 SQL 語(yǔ)句的模糊性而導(dǎo)致的潛在錯(cuò)誤。
2. 為什么需要 ONLY_FULL_GROUP_BY?
在 ONLY_FULL_GROUP_BY 被引入之前,MySQL 允許在沒(méi)有 GROUP BY 或者 GROUP BY 不充分的情況下進(jìn)行查詢(xún)。這意味著,即使查詢(xún)中包含了沒(méi)有被聚合的列,MySQL 也會(huì)返回任意一個(gè)結(jié)果,這可能會(huì)導(dǎo)致誤導(dǎo)性的結(jié)果或數(shù)據(jù)丟失。
例如,假設(shè)我們有以下查詢(xún):
SELECT prod_name, COUNT(order_num)
FROM products p
JOIN orders o ON p.prod_id = o.prod_id;
這里,prod_name 列沒(méi)有被聚合函數(shù)包裹,也沒(méi)有在 GROUP BY 子句中出現(xiàn)。在 ONLY_FULL_GROUP_BY 模式下,這個(gè)查詢(xún)會(huì)失敗,因?yàn)?MySQL 不知道如何從多個(gè)可能的 prod_name 值中選擇一個(gè)來(lái)展示。
3.如何啟用或禁用 ONLY_FULL_GROUP_BY?
在 MySQL 中,默認(rèn)情況下 ONLY_FULL_GROUP_BY 是啟用的。你可以通過(guò)檢查 @@sql_mode 系統(tǒng)變量來(lái)確認(rèn)這一點(diǎn):
SELECT @@sql_mode;
如果 ONLY_FULL_GROUP_BY 已經(jīng)被啟用,并且你想要暫時(shí)禁用它(盡管這不是一個(gè)推薦的長(zhǎng)期解決方案),你可以通過(guò)以下命令在會(huì)話(huà)級(jí)禁用它:
SET SESSION sql_mode=(SELECT REPLACE(@@sql_mode,'ONLY_FULL_GROUP_BY',''));
或者,你也可以在 MySQL 配置文件(my.cnf 或 my.ini)中修改 sql_mode 設(shè)置來(lái)全局禁用它。
總結(jié)
雖然禁用 ONLY_FULL_GROUP_BY 可能會(huì)方便一些查詢(xún)的編寫(xiě),但從長(zhǎng)遠(yuǎn)來(lái)看,遵循這個(gè)規(guī)則對(duì)于保持?jǐn)?shù)據(jù)查詢(xún)的準(zhǔn)確性和一致性至關(guān)重要。在編寫(xiě) SQL 查詢(xún)時(shí),始終應(yīng)確保遵循 ONLY_FULL_GROUP_BY 的指導(dǎo)原則,以避免潛在的數(shù)據(jù)解釋錯(cuò)誤。
柚子快報(bào)邀請(qǐng)碼778899分享:【MySQL 的 ONLY
文章來(lái)源
本文內(nèi)容根據(jù)網(wǎng)絡(luò)資料整理,出于傳遞更多信息之目的,不代表金鑰匙跨境贊同其觀(guān)點(diǎn)和立場(chǎng)。
轉(zhuǎn)載請(qǐng)注明,如有侵權(quán),聯(lián)系刪除。