0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

sql優(yōu)化,一定要用好Explain

Android編程精選 ? 來源:稀土掘金 ? 作者:沸羊羊 ? 2022-03-18 09:50 ? 次閱讀

Part1前言

  • 環(huán)境:Mysql 8.0.21
  • Mysql 版本不同 explain 執(zhí)行結(jié)果會(huì)不相同

Part2Explain的作用?

EXPLAIN 語(yǔ)句提供有關(guān)MySQL如何執(zhí)行語(yǔ)句的信息。EXPLAIN 可以作用于SELECT、DELETE、INSERT、REPLACE和UPDATE語(yǔ)句

explainselect語(yǔ)句中使用的每個(gè)表返回一行信息。它按 mysql 在處理語(yǔ)句時(shí)讀取的順序列出輸出的表。mysql 使用嵌套循環(huán)連接方法解析所有連接,這意味著 mysql 從第一個(gè)表中讀取一行,然后在第二個(gè)表中、第三個(gè)表中找到匹配的行,依此類推。處理完所有表后,Mysql 輸出所選列,并在表列表中回溯,直到找到一個(gè)有更多匹配行的表。從該表中讀取下一行,并繼續(xù)處理下一個(gè)表。

explain 輸出列信息

aa0ad88a-9474-11ec-952b-dac502259ad0.png

1.1 id

這是查詢中 select 的序列號(hào),如果該行引用其他行的并集結(jié)果,則該行可以為 Null

id 列的值代表著表的執(zhí)行順序,一共分為三種情況:

1.1.1 id相同

EXPLAINSELECTs.*,t.*FROMstudents,teachert

aa29321c-9474-11ec-952b-dac502259ad0.png

可以看到,explain 命令為 select 標(biāo)識(shí)符語(yǔ)句中每個(gè)表生成了一行信息,其中 id 相同,代表著 兩個(gè)表的執(zhí)行順序從上到下,與 sql 中執(zhí)行的順序無(wú)關(guān)。

1.1.2 id不同

EXPLAINSELECT*FROMteachertWHEREid=(SELECTs.tidFROMstudentsWHEREs.id="2")

aa3dfb16-9474-11ec-952b-dac502259ad0.png

嵌套子查詢的兩個(gè)表的執(zhí)行順序是不同的,所以 explain 解析出的查詢信息 id 是不同的。其中,id 越大代表優(yōu)先級(jí)越大,越先被執(zhí)行。

1.1.3 id有相同有不同

EXPLAINSELECTs.id,s.NAME,c.id,c.NAMEFROMclassc,studentsWHEREs.tid=(SELECTt.idFROMteachertWHEREt.id="1")

aa5196ee-9474-11ec-952b-dac502259ad0.png

如果 id 有相同,可以認(rèn)為是一組,從結(jié)果集中顯示的順序從上往下執(zhí)行(與 SQL 中聲明順序無(wú)關(guān));id 值越大,優(yōu)先級(jí)越高,越先執(zhí)行。

1.2 select_type

select 的類型,所有的情況見下表:aa6b35cc-9474-11ec-952b-dac502259ad0.jpg

1.2.1 SIMPLE

簡(jiǎn)單查詢,查詢中不包含子查詢或者union等任何復(fù)雜查詢 [詳情見示例1.1.1]

1.2.2 PRIMARY

查詢中若包含任何復(fù)雜的子查詢,則最外層被標(biāo)記為 primary,俗稱:雞蛋殼。[詳情見示例1.1.2]

1.2.3 UNION

EXPLAINSELECT`NAME`FROMtb_employees_chinaUNIONSELECT`NAME`FROMtb_employees_usa

aa8b0988-9474-11ec-952b-dac502259ad0.jpgunion 之后的 select 被標(biāo)記為 union,而 union 前的select 被標(biāo)記為primary ;若 union 包含在 from 子句的子查詢中,外層 select 將被標(biāo)記為derived

1.2.4 UNION RESULT

兩種 union 合并的結(jié)果。[詳情見示例 1.2.3]

1.2.5 DEPENDENT UNION

EXPLAINSELECT`NAME`FROMtb_employees_chinaWHERE`NAME`IN(SELECT`NAME`FROMtb_employees_chinaUNIONSELECT`NAME`FROMtb_employees_usa)
aa998184-9474-11ec-952b-dac502259ad0.jpg

首先要滿足 UNION 的條件,以及 union 中的第二個(gè)或以后的 select ,依賴于外部查詢。

1.2.6 SUBQUERY

EXPLAINSELECT*FROMtb_employees_chinaWHEREid=(SELECTidFROMtb_employees_chinaWHERE`name`="lisi")
aaa9e2e0-9474-11ec-952b-dac502259ad0.jpg

子查詢中的第一個(gè) select 被標(biāo)識(shí)為 subquery

1.2.7 DEPENDENT SUBQUERY

在子查詢中的 select ,依賴于外部查詢 [詳情見示例1.2.4] 此示例中子查詢的第一個(gè) select 被標(biāo)識(shí)為 dependent subquery

1.2.8 DERIVED

在 from 子句中包含的子查詢被標(biāo)記為 derived ,mysql 會(huì)遞歸這些子查詢,把結(jié)果放在臨時(shí)表中(臨時(shí)表會(huì)增加系統(tǒng)負(fù)擔(dān),但有時(shí)不得不用)。

注:此實(shí)例中,mysql 環(huán)境為:5.7

EXPLAINSELECT*FROM(SELECTidFROMtb_coursetc)temp

aac1d2f6-9474-11ec-952b-dac502259ad0.jpg由于 Mysql 8.0 在 Mysql 5.7 基礎(chǔ)上做了優(yōu)化,上述實(shí)例在 8.0 版本環(huán)境 explain ,子查詢是不會(huì)被標(biāo)識(shí)為 derived 的,暫時(shí)沒有找到 mysql 8.0 derived 的實(shí)例。(后續(xù)發(fā)現(xiàn)會(huì)做補(bǔ)充)

1.2.9 DEPENDENT DERIVED

在 derived 的基礎(chǔ)上,依賴于外部查詢。

1.2.10 MATERIALIZED

注:此實(shí)例中,mysql 環(huán)境為:5.7

EXPLAINSELECT*FROMtb_classWHERE`NAME`IN(SELECT`NAME`FROMtb_class)

aad62c7e-9474-11ec-952b-dac502259ad0.jpg

將子查詢結(jié)果集中的記錄保存到臨時(shí)表的過程稱之為物化(Materialize)。那個(gè)存儲(chǔ)子查詢結(jié)果集的臨時(shí)表稱之為物化表

在查詢優(yōu)化器執(zhí)行包含子查詢的語(yǔ)句時(shí),選擇將子查詢物化之后與外層查詢進(jìn)行連接查詢時(shí),該子查詢會(huì)被標(biāo)識(shí)為 MATERIALIZED

執(zhí)行計(jì)劃的第三條記錄的 id 值為2,說明該條記錄最先被執(zhí)行,并且是個(gè)單表查詢,它被標(biāo)識(shí)為 MATERIALIZED ,查詢優(yōu)化器是要把子查詢先轉(zhuǎn)換為物化表。執(zhí)行計(jì)劃的第二條記錄,也就是id 為1,table 為的記錄,此條記錄就是 id 為 2 對(duì)應(yīng)的子查詢執(zhí)行之后產(chǎn)生的物化表,再將tb_class 與 該物化表(兩個(gè) id 為 1 的表)連接查詢。

1.2.11 UNCACHEABLE SUBQUERY

無(wú)法緩存其結(jié)果的子查詢,必須為外部查詢的每一行重新計(jì)算結(jié)果

1.2.12 UNCACHEABLE UNION

union 中第二個(gè)或以后的 select ,屬于不可緩存的子查詢

1.3 table

輸出行所引用的表的名稱,除表名稱外還有三種 case

  • :id值為 M 和 N 之間的并集 [詳情見示例1.2.4]
  • :id 值為 N 的派生表結(jié)果 [詳情見示例1.2.8]
  • :id 值為 N 的物化的子查詢的結(jié)果 [詳情見示例1.2.10]

1.4 type

官方全稱是join type,意為:連接類型。Mysql 8.0 中 type 類型達(dá)到了12種,下面著重介紹常用的 6 種。從上到下,效率依次是增強(qiáng)的,我們應(yīng)該盡量?jī)?yōu)化我們的 sql,使它的 type盡量更優(yōu),當(dāng)然還要綜合考慮實(shí)際情況。

1.4.1 all(Full Table Scan)

全盤掃描,對(duì)表中的每行組合都執(zhí)行一次完整的表掃描,如果表是第一個(gè)沒有標(biāo)記 const 的表,通??梢酝ㄟ^索引來避免 ALL,這些索引允許基于先前表中的常量值或列值從表中檢索行。

ALL 是一種暴力和原始的查找方法,非常耗時(shí)低效。但Mysql官方介紹了一些情況可以使用ALL掃描:

  • 該表很小,以至于執(zhí)行表掃描要比打擾key查找快得多,對(duì)于少于10行且行長(zhǎng)較短的表,是比較常見的
  • 對(duì)于索引列,ON 或者 WHERE 子句中沒有可用的限制
  • 正在將索引列與常量值進(jìn)行比較
  • 正在通過另一列使用基數(shù)較低的鍵(許多行與鍵值匹配),這種情況下,Mysql假定通過使用鍵可能需要進(jìn)行多次鍵查找,并且表掃描會(huì)更快。

對(duì)于小型表,表掃描通常是合適的,并且對(duì)于性能的影響可以忽略不計(jì),對(duì)于大型表,一定要進(jìn)行優(yōu)化查詢。

SELECT*FROM`tb_employees_china`WHERE`name`="zhangsan"

aaecdb2c-9474-11ec-952b-dac502259ad0.jpg這是因?yàn)?name 列既不是主鍵也沒有索引,所以采用全盤掃描的方式查找。

1.4.2 index(Full Index Scan)

indexall 都是全盤掃描,區(qū)別就是 index 掃描的是索引樹,這種掃描根據(jù)索引回表取數(shù)據(jù),和 all 相比,他們都是取得了全表的數(shù)據(jù),而且 index 要先讀索引而且要回表隨機(jī)取數(shù)據(jù),index 不會(huì)比 all 快。

SELECTidFROM`tb_employees_usa`
aafd6190-9474-11ec-952b-dac502259ad0.jpg

而如果 type 為 index,并且 Extra 為 Using index ,如上圖這種情況,就使用了覆蓋索引,也就是無(wú)需回表,當(dāng)前的索引樹滿足了當(dāng)前的查詢需求。

1.4.3 range

range指的是有范圍的索引掃描,相對(duì)于 index的索引掃描,它有范圍限制,因此要優(yōu)于 index,range一定是基于索引的,一般常見的范圍查找:between...and,<,>,inor 都屬于索引范圍掃描。

1.4.4 ref

出現(xiàn)該連接類型的條件是,查找條件列使用了索引而且不為主鍵和 Unique,也就是使用了普通索引,而非主鍵索引和唯一索引。這樣,即使使用索引快速查找到了第一條數(shù)據(jù),也不能停止掃描,要進(jìn)行目標(biāo)值附近的小范圍掃描,好處是不需要掃全表,因?yàn)樗饕怯行虻模词褂兄貜?fù)值,也是一個(gè)非常小的范圍內(nèi)掃描。

將以下表中 tb_employees_usaname 字段新建索引后執(zhí)行以下語(yǔ)句。

explainselect*fromtb_employees_usawhere`name`='rose';
ab0e7552-9474-11ec-952b-dac502259ad0.jpg

1.4.5 eq_ref

eq_refref 相比厲害的地方在于,eq_ref 知道這種類型的查找結(jié)果集只有一個(gè),只有使用了主鍵或者唯一索引進(jìn)行查找的情況,結(jié)果集才會(huì)是一個(gè)。在查找前就已經(jīng)知道結(jié)果集一定只有一個(gè),所以,當(dāng)首次查找到值時(shí),就立即停止了查詢。這種連接類型每次都進(jìn)行著精確查詢,無(wú)需過多的掃描,因此查找效率更高。

select*fromtb_employees_usajointb_employees_chinausing(id);
ab2313f4-9474-11ec-952b-dac502259ad0.jpg

1.4.6 const

通常情況下,如果將一個(gè)主鍵放置到 where 子句作為查詢條件,MySQL 優(yōu)化器會(huì)把這個(gè)查詢優(yōu)化為一個(gè)常量,也就是 const

select*fromtb_employees_usawhere`id`='3';
ab3207ec-9474-11ec-952b-dac502259ad0.jpg

1.5 possible_keys

查詢可能使用到的索引都會(huì)在這里列出來

1.6 key

查詢真正使用到的索引

1.7 key_len

key_len 表示使用的索引長(zhǎng)度,key_len可以衡量索引的好壞,key_len越小索引效果越好,那么 key_len 長(zhǎng)度是如何計(jì)算的呢?

列類型 是否為空 長(zhǎng)度 key_len 備注
tinyint 允許Null 1 key_len = 1+1 允許NULL,key_len長(zhǎng)度加1
tinyint 不允許Null 1 key_len = 1 不允許NULL
int 允許Null 4 key_len = 4+1 允許NULL,key_len長(zhǎng)度加1
int 不允許Null 4 key_len = 4 不允許NULL
bigint 允許Null 8 key_len = 8+1 允許NULL,key_len長(zhǎng)度加1
bigint 不允許Null 8 key_len = 8 不允許NULL
char(1) 允許Null utf8mb4=4,utf8=3,gbk=2 key_len = 1*3 + 1 允許NULL,字符集utf8,key_len長(zhǎng)度加1
char(1) 不允許Null utf8mb4=4,utf8=3,gbk=2 key_len = 1*3 不允許NULL,字符集utf8
varchar(10) 允許Null utf8mb4=4,utf8=3,gbk=2 key_len = 10*3 + 2 + 1 動(dòng)態(tài)列類型,key_len長(zhǎng)度加2,允許NULL,key_len長(zhǎng)度加1
varchar(10) 不允許Null utf8mb4=4,utf8=3,gbk=2 key_len = 10*3 + 2 動(dòng)態(tài)列類型,key_len長(zhǎng)度加2

拿我們上述的執(zhí)行計(jì)劃來看,計(jì)算一個(gè)索引的長(zhǎng)度。ab5510b6-9474-11ec-952b-dac502259ad0.jpg其中,索引名稱為 idx_name 的索引類型是 varchar(30),根據(jù)上述圖表得知,varchar(30) key_len 為:30*3+2+1 = 93

key_len還可用于判斷聯(lián)合索引是否生效以及覆蓋了哪個(gè)聯(lián)合索引,我們舉個(gè)實(shí)例來看。

我們將 tb_employees_usa 表的索引全部刪除,新建聯(lián)合索引 (name,age)ab6bcedc-9474-11ec-952b-dac502259ad0.jpg

執(zhí)行以下SQL,查看執(zhí)行計(jì)劃

explainselect*fromtb_employees_usawherename='rose';

ab8677aa-9474-11ec-952b-dac502259ad0.jpgkey_len 值為93,經(jīng)計(jì)算,此 SQL 使用了聯(lián)合索引其中的單個(gè)索引 name

再執(zhí)行如下SQL,查看執(zhí)行計(jì)劃

explainselect*fromtb_employees_usawherename='rose';

ab9f9096-9474-11ec-952b-dac502259ad0.jpgkey_len 值為98,經(jīng)計(jì)算,此 SQL 使用了聯(lián)合索引。

1.8 ref

ref字段的值是列或者常數(shù),指的是這個(gè)列或常數(shù)與 key 的值一起從表中選擇行數(shù)據(jù)。如上述實(shí)例

select*fromtb_employees_usajointb_employees_chinausing(id);

ab2313f4-9474-11ec-952b-dac502259ad0.jpgref 字段值為 test.tb_employees_usa.id,代表 tb_employees_usaid 列與 tb_employees_china 的主鍵一起篩選行數(shù)據(jù)。

where 子句中條件是等值常量,則 ref 值為 const

1.9 rows

用來表示在SQL執(zhí)行過程中會(huì)被掃描的行數(shù),該數(shù)值越大,意味著需要掃描的行數(shù),相應(yīng)的耗時(shí)更長(zhǎng)。但是需要注意的是EXPLAIN 中輸出的 rows 只是一個(gè)估算值,不能完全對(duì)其百分之百相信。

1.10 filtered

表示存儲(chǔ)引擎返回的數(shù)據(jù)在 server 層過濾后,剩下多少滿足查詢的記錄數(shù)量的比例,此值是百分比,不是具體記錄數(shù)量。

1.11 Extra

性能從好到壞排列:

usingindex>usingwhere>usingtemporary>usingfilesort
  • using index :表示覆蓋索引,不需要回表操作
  • using where:列數(shù)據(jù)是從僅僅使用了
  • using temporary:表示 MySQL 需要使用臨時(shí)表來存儲(chǔ)結(jié)果集,常見于排序和分組查詢
  • using filesort:MySQL 中無(wú)法利用索引完成的排序操作稱為“文件排序”,一般有此值建議使用索引進(jìn)行優(yōu)化
  • using join buffer:強(qiáng)調(diào)了在獲取連接條件時(shí)沒有使用索引,并且需要連接緩沖區(qū)來存儲(chǔ)中間結(jié)果,如果出現(xiàn)此值,應(yīng)該根據(jù)具體情況添加索引來改進(jìn)性能
  • distinct:在select部分使用了distinc關(guān)鍵字

-End-

審核編輯 :李倩



聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • SQL
    SQL
    +關(guān)注

    關(guān)注

    1

    文章

    772

    瀏覽量

    44206
  • MySQL
    +關(guān)注

    關(guān)注

    1

    文章

    826

    瀏覽量

    26664

原文標(biāo)題:sql優(yōu)化,你一定要用好Explain

文章出處:【微信號(hào):AndroidPush,微信公眾號(hào):Android編程精選】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏

    評(píng)論

    相關(guān)推薦

    ARM寫邏輯程序是不是一定要用到下載器?

    ARM寫邏輯程序是不是一定要用到下載器?
    發(fā)表于 04-27 15:18

    宏定義下的“GPIO_”是什么作用,一定要用

    #defineGPIO_KEY P1//獨(dú)立鍵盤用P1口#defineGPIO_LED P0//led使用P0口我想知道這個(gè)宏定義下的“GPIO_”是什么作用,一定要用嗎?c程序中的GPIO_與通訊輸入\輸出的GPIO是同個(gè)嗎?
    發(fā)表于 05-29 12:22

    為什么第盞燈一定要用0xfe,其他的燈也一定要用該數(shù)值才有效?

    為什么第盞燈一定要用0xfe,其他的燈也一定要用該數(shù)值才有效?因?yàn)榻佑|這個(gè)時(shí)間不多,請(qǐng)你們盡量教會(huì)我啊 ,謝謝雖然你們都會(huì)說這很簡(jiǎn)單,但我還是不懂,請(qǐng)你們耐心指導(dǎo)下小白的我啊
    發(fā)表于 10-18 22:46

    請(qǐng)問飛控主機(jī)是不是一定要用操作系統(tǒng)?

    飛控主機(jī)是不是一定要用操作系統(tǒng),可以單片機(jī)直接跑嗎?
    發(fā)表于 07-24 23:19

    SQL語(yǔ)句怎么優(yōu)化

    SQL語(yǔ)句優(yōu)化——結(jié)合書籍論壇小結(jié)
    發(fā)表于 06-14 14:46

    ARM寫邏輯程序一定要用到下載器嗎?

    ARM寫邏輯程序是不是一定要用到下載器?
    發(fā)表于 07-16 08:10

    請(qǐng)問充電器一定要用y電容嗎?

    充電器一定要用y電容嗎?
    發(fā)表于 10-17 07:18

    單片機(jī)一定要用電腦燒錄程序嗎?

    單片機(jī)一定要用電腦燒錄程序嗎,可以用其他設(shè)備,比如手機(jī)嗎?
    發(fā)表于 10-20 06:29

    全網(wǎng)最全面、最細(xì)致的EXPLAIN解讀

    )。 所以,我肝了將近個(gè)星期,整理了下。這應(yīng)該是全網(wǎng)最全面、最細(xì)致的EXPLAIN解讀文章了,下面是全文。 文章比較長(zhǎng),建議收藏。 TIPS 本文基于MySQL 8.0編寫,理論支持MySQL 5.0
    的頭像 發(fā)表于 10-30 16:39 ?2406次閱讀

    SQL后悔藥,SQL性能優(yōu)化SQL規(guī)范優(yōu)雅

    個(gè)好習(xí)慣都是筆財(cái)富,本文基于MySQL,分SQL后悔藥, SQL性能優(yōu)化,SQL規(guī)范優(yōu)雅三
    的頭像 發(fā)表于 11-14 09:54 ?1866次閱讀

    5G基站測(cè)試為什么一定要用終端仿真器?資料下載

    電子發(fā)燒友網(wǎng)為你提供5G基站測(cè)試為什么一定要用終端仿真器?資料下載的電子資料下載,更有其他相關(guān)的電路圖、源代碼、課件教程、中文資料、英文資料、參考設(shè)計(jì)、用戶指南、解決方案等資料,希望可以幫助到廣大的電子工程師們。
    發(fā)表于 04-07 08:52 ?12次下載
    5G基站測(cè)試為什么<b class='flag-5'>一定要用</b>終端仿真器?資料下載

    如何通過explain來驗(yàn)證sql的執(zhí)行順序

    by 先執(zhí)行,到底它倆誰(shuí)先執(zhí)行呢? 今天我們通過 explain 來驗(yàn)證下 sql 的執(zhí)行順序。 在驗(yàn)證之前,先說結(jié)論,Hive 中 sql 語(yǔ)句的執(zhí)行順序如下: from 。. where
    的頭像 發(fā)表于 09-07 16:24 ?2703次閱讀
    如何通過<b class='flag-5'>explain</b>來驗(yàn)證<b class='flag-5'>sql</b>的執(zhí)行順序

    電路設(shè)計(jì)降壓,一定要用變壓器嗎?

    降壓,一定要用變壓器嗎?其實(shí)不是這樣的,除了變壓器,電容也是可以降壓,今天就帶領(lǐng)大家了解下電容降壓!
    發(fā)表于 02-10 10:08 ?4次下載
    電路設(shè)計(jì)降壓,<b class='flag-5'>一定要用</b>變壓器嗎?

    sql優(yōu)化常用的幾種方法

    前言 1.慢SQL優(yōu)化思路。 1.1 慢查詢?nèi)罩居涗浡?b class='flag-5'>SQL 1.2 explain查看分析SQL的執(zhí)行計(jì)劃 1.3 profile 分析執(zhí)
    的頭像 發(fā)表于 11-14 15:04 ?5194次閱讀

    丙類放大器為什么一定要用調(diào)諧回路作為集電極負(fù)載?

    丙類放大器為什么一定要用調(diào)諧回路作為集電極負(fù)載?回路為什么一定要調(diào)到諧振狀態(tài)?回路失諧將產(chǎn)生什么結(jié)果? 丙類放大器是種功率放大器,其中集電極是其關(guān)鍵部分。它通常需要個(gè)調(diào)諧回路作為集
    的頭像 發(fā)表于 10-11 17:43 ?1862次閱讀