色综合图-色综合图片-色综合图片二区150p-色综合图区-玖玖国产精品视频-玖玖香蕉视频

您的位置:首頁技術(shù)文章
文章詳情頁

MySQL 分組查詢的優(yōu)化方法

瀏覽:3日期:2023-10-03 11:42:59

MySQL 在處理 GROUP BY 和 DISTINCT 查詢的方式在大多數(shù)情況下類似,事實(shí)上,在優(yōu)化過程中有時候會把在這兩種方式中轉(zhuǎn)換。兩類查詢都能夠從索引中受益,通常,這也是優(yōu)化這兩種查詢最為重要的方式。

在無法使用索引時,MySQL 對于 GROUP BY 查詢有兩種策略:使用臨時表或者 filesort 執(zhí)行分組。對于給定的查詢,兩種方式都沒法更高效。我們可以通過配置 SQL_BIG_RESULT 和 SQL_SMALL_RESULT 來指定優(yōu)化器選擇其中一個方式。

通常,對查詢表的id 進(jìn)行分組比使用值進(jìn)行分組效率更高,例如下面的查詢效率就比較低:

SELECT actor.first_name, actor.last_name, COUNT(*)FROM sakila.file_actorINNER JOIN sakila.actor USING(actor_id)GROUP BY actor.first_name, actor.last_name;

而下面的查詢方式則更有效:

SELECT actor.first_name, actor.last_name, COUNT(*)FROM sakila.file_actorINNER JOIN sakila.actor USING(actor_id)GROUP BY film_actor.actor_id;

而使用 actor.actor_id 進(jìn)行分組會比 film_actor.actor_id更高效。

這個查詢能夠發(fā)揮其優(yōu)勢的依據(jù)是演員(actor)的姓名是依賴于 actor_id 的,因此會返回相同的結(jié)果,但是如果返回的結(jié)果不同的話就不能這么做了。甚至有些時候服務(wù)端通過 SQL_MODE 配置禁用了 GROUP BY。此時如果不關(guān)心獲取的值,而且用于分組的列的值是唯一的,這可以使用 MIN和 MAX 來解決這個問題。

SELECT MIN(actor.first_name), MAX(actor.last_name), ...;

對于完美主義者,他們會認(rèn)為你的分組是錯誤的,他們也是對的。一個虛擬的 MIN 或 MAX 的結(jié)果是查詢并不會正確地組裝。然而,有時候你只是為了讓 MySQL 更快地執(zhí)行查詢。完美主義者對于下面的查詢會滿意:

SELECT actor.fisrt_name, actor.last_name, c.cntFROM sakila.actorINNER JOIN ( SELECT actor_id, COUNT(*) AS cnt FROM sakila.film_actor GROUP BY actor_id ) AS c USING(actor_id);

然而,子查詢中創(chuàng)建和填充臨時表的代價可能比理論上看起來的死辦法更高。需要記住的是,子查詢構(gòu)建的臨時表是沒有索引的,這會導(dǎo)致性能上的下降。

通常在分組查詢中,選擇沒有分組的列是一個糟糕的主意。這是因?yàn)椴樵兘Y(jié)果是不確定的,一旦改變了索引或優(yōu)化器使用了不同的策略都會導(dǎo)致結(jié)果被改變。事實(shí)上,我們建議將服務(wù)端的 SQL_MODE 設(shè)置為 ONLY_FULL_GROUP_BY,這時寫了一個糟糕的分組查詢時,系統(tǒng)會產(chǎn)生一個錯誤而不是直接執(zhí)行。開啟 ONLY_FULL_GROUP_BY 后,SELECT 的字段只能是 GROUP BY 指定的字段,此時可以通過構(gòu)建分步查詢或子查詢的方式,先分組查出分組的列,再做二次查詢。

MySQL 會根據(jù) GROUP BY 指定的列次序自動分組,除非是使用了 ORDER BY 指定排序規(guī)則。如果不在乎次序并且發(fā)現(xiàn)了這導(dǎo)致了一個 filesort,這時候可以使用 ORDER BY NULL 來跳過自動排序。也可以通過在 GROUP BY 后面增加 DESC 或 ASC 來指定結(jié)果按指定的方向排序。

有時候可以在分組查詢時要求 MySQL 在結(jié)果中做一次超級聚合。這可以通過在 GROUP BY 后面增加WITH ROLLUP 子句完成,但是這不一定能夠達(dá)到優(yōu)化的預(yù)期。可以通過 EXPLAIN 檢查執(zhí)行的方法,注意分組有沒有通過 filesort 或臨時表完成。然后在對相同的查詢移除 WITH ROLLUP 后進(jìn)行對比。通過對比也許可以找到優(yōu)化的辦法。

有些時候通過增加聚合查詢會使得效率更高,雖然這種方式會返回更多的行。也可以通過在 FROM 后面嵌套子查詢來保持中間查詢結(jié)果,然后再使用 UNION 獲取最終結(jié)果。

但是注意的是,在應(yīng)用程序中最好是移除 WITH ROLLUP,而通過優(yōu)化來完成分組查詢。

結(jié)語:使用 GROUP BY 進(jìn)行分組查詢時最好是使用索引列分組,若無需指定次序可以使用 ORDER BY NULL 進(jìn)行優(yōu)化。倘若不按索引列分組的時候,則需要考慮變通的辦法,并且考慮是否要使用子查詢或使用 WITH ROLLUP 檢查性能后再做優(yōu)化。同時,為了防止分組查詢出現(xiàn)不可預(yù)料的錯誤,最好是開啟 ONLY_FULL_GROUP_BY。

以上就是MySQL 分組查詢的優(yōu)化方法的詳細(xì)內(nèi)容,更多關(guān)于MySQL 分組查詢的優(yōu)化的資料請關(guān)注好吧啦網(wǎng)其它相關(guān)文章!

標(biāo)簽: MySQL 數(shù)據(jù)庫
相關(guān)文章:
主站蜘蛛池模板: 久久免费国产视频 | 日本加勒比视频 | 国产碰碰| 国产免费一区二区三区在线观看 | 国产黄页 | 成人爽a毛片在线视频 | 欧美另类精品一区二区三区 | 真实一级一级一片免费视频 | 亚洲最大福利视频 | 美国免费一级片 | 欧美一级永久免费毛片在线 | aa级毛片毛片免费观看久 | 六月丁香婷婷色狠狠久久 | 亚洲一区二区精品推荐 | 亚洲成人播放 | 99re最新这里只有精品 | 国产一级一片免费播放视频 | 欧美日韩高清观看一区二区 | 亚洲精品一区亚洲精品 | 久久这里有精品视频 | 亚洲国产精品二区久久 | 久久免费高清 | 美女黄网站色一级毛片 | 久久精品一区二区三区不卡牛牛 | 中国美女乱淫免费看视频 | 国产精品亚洲专一区二区三区 | 亚洲在线视频免费 | 免费看亚洲 | 国产自产自拍 | 米奇777色狠狠8888影视 | 免费v片视频在线观看视频 免费v片在线观看 | avove旗袍丝袜高跟啪啪 | 欧美色v| 久久国内精品自在自线观看 | 视色4setv.com | 成人亚洲综合 | 夜夜春夜夜夜夜猛噜噜噜噜噜 | 99色视频在线 | 欧美综合一区 | 亚洲日本久久一区二区va | 成人精品国产 |