欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

MySQL order by與group by查詢優(yōu)化實(shí)現(xiàn)詳解

 更新時(shí)間:2022年11月01日 11:20:08   作者:流煙默  
order by 子句盡量使用index方式排序(即using index),避免使用filesort方式排序(即using filesort)。Index方式效率高,它指MySQL掃描索引本身完成排序,filesort則效率低

前言

order by滿足兩種情況,會(huì)使用 index 方式排序:

  • order by語句使用索引最左前列(最左匹配法則)
  • where子句和order by子句條件列組合滿足最左匹配法則(where條件使用索引的最左前綴為常量)

下面給出幾個(gè)實(shí)例來說明,如下所示我們創(chuàng)建表并為其創(chuàng)建組合索引(c1,c2,c3)。

CREATE TABLE `testc` (
  `id` bigint(20) NOT NULL AUTO_INCREMENT,
  `c1` varchar(100) DEFAULT NULL,
  `c2` varchar(100) DEFAULT NULL,
  `c3` varchar(100) DEFAULT NULL,
  `c4` varchar(100) DEFAULT NULL,
  `c5` varchar(100) DEFAULT NULL,
  PRIMARY KEY (`id`),
  KEY `testc_c1_IDX` (`c1`,`c2`,`c3`) USING BTREE
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4

where與order by滿足最左匹配法則

# c1 c2滿足最左匹配法則
explain select * from testc where c1='a1' order by c2
# 與上面等價(jià)
explain select * from testc where c1='a1' order by c2,c3

key_len標(biāo)明查找用到了索引 c1,Extra中是Using index condition 沒有同時(shí)出現(xiàn)using where ,表明 c2 索引用來讀取數(shù)據(jù)而非執(zhí)行查找動(dòng)作。

MySQL Innodb下的B+樹本身就是多路平衡樹,那么索引換句話就是排好序的快速查找數(shù)據(jù)結(jié)構(gòu)。如果order by用到了索引且排序和索引次序一樣,那么無疑效果是最好的。

中間斷裂

如下所示,缺少了c2,order by不滿足最左匹配法則。

explain select * from testc where c1='a1' order by c3

可以看到Extra中Using index condition; Using filesort說明雖然where可以用到索引(單獨(dú)c1滿足最左匹配),但是排序不滿足,故而出現(xiàn)了filesort。

大哥不在

如下c1不在,那么很顯然無論查找還是排序都用不到索引。

explain select * from testc where c2='a2' order by c3

這里Extra是Using where; Using filesort,說明通過where子句過濾結(jié)果,然后對(duì)結(jié)果進(jìn)行文件排序。

范圍失效

如下所示,中間c2是個(gè)范圍搜索,那么其后索引將失效也就是order by c3無法與where連接滿足最左匹配法則。

explain select * from testc where c1='a1' and c2 > 'a2' order by c3

如下圖所示,這里type = range,ken_len表示用到了 c1,c2索引。Extra是Using index condition; Using filesort表示查詢用到了索引但是無法利用索引完成的排序操作。

這種情況如何優(yōu)化呢?order by c2,c3!這樣就可以保證索引排序而不需要filesort。

explain select * from agriculture.testc where c1='a1' and c2  > 'a2' 
order by  c2,c3

order by 次序相反

如下所示,order by的次序沒有與索引次序保持一致。這里Extra為Using index condition; Using filesort。

explain select * from testc where c1='a1' order by c3,c2

覆蓋索引

前面幾個(gè)都是select *,這里查找索引列。

沒有where,order by滿足全值匹配,select查詢的數(shù)據(jù)是索引列。

explain select c1 from testc order by c1, c2,c3

這里Extra中只有Using index;

沒有where,order by 大哥丟失,select查詢的數(shù)據(jù)是索引列。

explain select c1 from testc order by c2,c3

這里Extra中是Using index; Using filesort 。

這里Extra信息為Using where; Using index; Using filesort

explain select c1 from testc where c1='a1' order by c3,c2

filesort的兩種算法

filesort有兩種機(jī)制:雙路排序和單路排序。雙路排序簡(jiǎn)單來講就是兩次掃描磁盤,最終得到數(shù)據(jù)。單路排序則是只需要讀取一次,也就是一次磁盤IO。

雙路排序

MySQL4.1之前是使用雙路排序,讀取行指針和order by列,對(duì)他們進(jìn)行排序,然后掃描已經(jīng)排序好的列表,按照列表中的值重新從列表中讀取對(duì)應(yīng)的數(shù)據(jù)輸出(可以理解為從磁盤讀取排序字段,在buffer進(jìn)行排序,然后再?gòu)拇疟P讀取其他字段)。

取一批數(shù)據(jù)要進(jìn)行兩次磁盤IO,這是很耗時(shí)的。故而在MySQL4.1之后,出現(xiàn)了第二種改進(jìn)的算法,也就是單路排序。

單路排序

從磁盤讀取查詢需要的所有列,按照order by列在buffer對(duì)它們進(jìn)行排序,然后掃描排序后的列表進(jìn)行輸出。它的效率更快一點(diǎn),避免了第二次讀取數(shù)據(jù),并且把隨機(jī)IO變成了順序IO。但是其會(huì)使用更多的空間,因?yàn)槠渚彺媪藬?shù)據(jù)在內(nèi)存中。

單路的問題

可能取出的數(shù)據(jù)大小超過了sort_buffer的容量,導(dǎo)致每次只能取sort_buffer容量大小的數(shù)據(jù)進(jìn)行排序(創(chuàng)建tmp文件,多路合并),排完再取sort_buffer容量大小…從而多次IO(可能比雙路更多)。

可以嘗試增大sort_buffer_size參數(shù)的設(shè)置或者max_length_for_sort_data參數(shù)的設(shè)置。

總結(jié)

order by時(shí)select * 是一個(gè)大忌,應(yīng)該是查詢需要的字段。

當(dāng)query的字段大小總和小于max_length_for_sort_data而且排序字段不是text|blob類型時(shí),會(huì)用改進(jìn)后的算法–單路排序,否則使用雙路排序。

兩種算法的數(shù)據(jù)都有可能超出sort_buffer的容量,超出之后會(huì)創(chuàng)建tmp文件進(jìn)行合并排序?qū)е露啻蜪O。尤其對(duì)于單路排序來說風(fēng)險(xiǎn)更大,所以需要適當(dāng)調(diào)整sort_buffer的容量。

提高max_length_for_sort_data會(huì)增加使用單路排序算法的概率。但是如果設(shè)置的太高,數(shù)據(jù)總?cè)萘砍^sort_buffer的概率就增大,明顯癥狀是磁盤IO高,CPU使用率低。

group by

前面提到的規(guī)則針對(duì)group by均適用,group by 實(shí)質(zhì)是先排序后分組,遵照索引建的最佳左前綴。當(dāng)無法使用索引時(shí),增大max_length_for_sort_data和sort_buffer參數(shù)的值。

需要注意的是where優(yōu)先級(jí)高于having,能寫在where限定的條件盡量不要通過having。

到此這篇關(guān)于MySQL order by與group by查詢優(yōu)化實(shí)現(xiàn)詳解的文章就介紹到這了,更多相關(guān)MySQL order by與group by內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • MySQL 5.7增強(qiáng)版Semisync Replication性能優(yōu)化

    MySQL 5.7增強(qiáng)版Semisync Replication性能優(yōu)化

    這篇文章主要介紹了MySQL 5.7增強(qiáng)版Semisync Replication性能優(yōu)化,本文著重講解支持發(fā)送binlog和接受ack的異步化、支持在事務(wù)commit前等待ACK兩項(xiàng)內(nèi)容,需要的朋友可以參考下
    2015-05-05
  • MySQL OOM 系列一 Linux內(nèi)存分配

    MySQL OOM 系列一 Linux內(nèi)存分配

    今天想提到的是線上一個(gè)4G的RDS實(shí)例,發(fā)生了OOM(out of memory)的問題,MySQL進(jìn)程被直接Kill掉了。在解釋這個(gè)問題的時(shí)候,我們首先需要從Linux系統(tǒng)內(nèi)存分配策略講起
    2016-07-07
  • mysql索引基數(shù)概念與用法示例

    mysql索引基數(shù)概念與用法示例

    這篇文章主要介紹了mysql索引基數(shù)概念與用法,結(jié)合實(shí)例形式分析了mysql索引基數(shù)的相關(guān)概念、原理、操作命令及相關(guān)使用技巧,需要的朋友可以參考下
    2019-03-03
  • MySQL中表的幾種連接方式

    MySQL中表的幾種連接方式

    這篇文章主要給大家介紹了關(guān)于MySQL中表的幾種連接方式,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-11-11
  • MySQL忽略表名大小寫的2種方法實(shí)現(xiàn)

    MySQL忽略表名大小寫的2種方法實(shí)現(xiàn)

    在 MySQL 中,默認(rèn)情況下表名是大小寫敏感的,本文主要介紹了MySQL忽略表名大小寫的2種方法實(shí)現(xiàn),具有一定的參考價(jià)值,感興趣的可以了解一下
    2024-03-03
  • 圖文詳解Mysql中如何查看Sql語句的執(zhí)行時(shí)間

    圖文詳解Mysql中如何查看Sql語句的執(zhí)行時(shí)間

    寫程序的人往往需要分析所寫的SQL語句是否已經(jīng)優(yōu)化過了,服務(wù)器的響應(yīng)時(shí)間有多快,所以下面這篇文章主要給大家介紹了關(guān)于Mysql中如何查看Sql語句的執(zhí)行時(shí)間的相關(guān)資料,需要的朋友可以參考下
    2021-12-12
  • MySQL WorkBench管理操作MySQL教程

    MySQL WorkBench管理操作MySQL教程

    MySQL Workbench提供DBAs和developers一個(gè)集成工具環(huán)境,方便管理mysql數(shù)據(jù)庫(kù),這里簡(jiǎn)單介紹下MySQL Workbench使用方法,需要的朋友可以參考下
    2014-03-03
  • MySQL中“:=”和“=”的區(qū)別淺析

    MySQL中“:=”和“=”的區(qū)別淺析

    這篇文章主要給大家介紹了關(guān)于MySQL中":="和"="區(qū)別的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者使用MySQL具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-08-08
  • MySQL安裝后不能用是什么情況該如何解決

    MySQL安裝后不能用是什么情況該如何解決

    之前安裝過MYSQL好像不用手動(dòng)啟動(dòng)服務(wù),具體也忘記了,但我上回給公司安裝的那個(gè)是要手動(dòng)安裝服務(wù)的,如果mysql剛剛安裝不能用,可能是服務(wù)沒有安裝
    2014-03-03
  • 如何修改Linux服務(wù)器中的MySQL數(shù)據(jù)庫(kù)密碼

    如何修改Linux服務(wù)器中的MySQL數(shù)據(jù)庫(kù)密碼

    這篇文章主要介紹了如何修改Linux服務(wù)器中的MySQL數(shù)據(jù)庫(kù)密碼問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-06-06

最新評(píng)論