久久久久久久视色,久久电影免费精品,中文亚洲欧美乱码在线观看,在线免费播放AV片

<center id="vfaef"><input id="vfaef"><table id="vfaef"></table></input></center>

    <p id="vfaef"><kbd id="vfaef"></kbd></p>

    
    
    <pre id="vfaef"><u id="vfaef"></u></pre>

      <thead id="vfaef"><input id="vfaef"></input></thead>

    1. 站長資訊網(wǎng)
      最全最豐富的資訊網(wǎng)站

      完全掌握Mysql的explain

      本篇文章給大家?guī)砹岁P(guān)于mysql的相關(guān)知識,其中主要介紹了explain的相關(guān)問題,Mysql中的explain堪稱Mysql的性能優(yōu)化分析神器,我們可以通過它來分析SQL語句的對應(yīng)的執(zhí)行的,希望對大家有幫助。

      完全掌握Mysql的explain

      推薦學(xué)習(xí):mysql視頻教程

      數(shù)據(jù)庫性能優(yōu)化是每個(gè)后端程序猿必備的基礎(chǔ)技能之一,而Mysql中的explain堪稱Mysql的性能優(yōu)化分析神器,我們可以通過它來分析SQL語句的對應(yīng)的執(zhí)行計(jì)劃在Mysql底層到底是如何執(zhí)行的,它對于我們評估SQL的執(zhí)行效率以及確定Mysql的性能優(yōu)化方向具有重要的意義。但是很多同學(xué)對于如何根據(jù)explain對已有SQL進(jìn)行深度的執(zhí)行分析還是丈二和尚摸不著頭腦,因此本文詳細(xì)闡述通過explain分析定位數(shù)據(jù)庫性能問題。

      explain基礎(chǔ)

      對于每個(gè)SQL來說,當(dāng)它被客戶端發(fā)送到Mysql服務(wù)端之后,會(huì)經(jīng)過Mysql的優(yōu)化器部件的分析,主要包括一些特殊的處理、執(zhí)行順序的改變以確保最優(yōu)的執(zhí)行效率,最終生成對應(yīng)的執(zhí)行計(jì)劃。所謂的執(zhí)行計(jì)劃,實(shí)際就是在存儲引擎層面如何獲取數(shù)據(jù)的,是通過索引獲取數(shù)據(jù)還是進(jìn)行全表掃描獲取數(shù)據(jù),獲取到數(shù)據(jù)后需不需要回表,等等,簡單理解就是Mysql獲取數(shù)據(jù)的過程。

      接下來我們來詳細(xì)看下,這個(gè)explain到底是何方神圣,為什么能指導(dǎo)我們進(jìn)行性能優(yōu)化。當(dāng)我們執(zhí)行如下語句:

      explain SELECT * FROM user_info where NAME='mufeng'

      執(zhí)行explain語句之后,我們會(huì)得到如下的執(zhí)行結(jié)果,這個(gè)類似數(shù)據(jù)庫表的12個(gè)字段實(shí)際上就是對Mysql執(zhí)行怎樣的執(zhí)行計(jì)劃的詳細(xì)描述。下面我們來好好研究下這12個(gè)字段分別代表什么意思,只有搞清楚它們的含義,我們才能明確Mysql到底是怎么執(zhí)行數(shù)據(jù)查詢的。

      完全掌握Mysql的explain

      完全掌握Mysql的explain

      1、id

      實(shí)際上每次select查詢都會(huì)對應(yīng)一個(gè)id,它代表著SQL執(zhí)行的順序,如果id值越大,說明對應(yīng)的SQL語句執(zhí)行的優(yōu)先級越高。在一些復(fù)雜的查詢SQL語句中常常包含一些子查詢,那么id序號就會(huì)遞增,如果出現(xiàn)嵌套查詢,我們可以發(fā)現(xiàn)最里層的查詢對應(yīng)的id最大,因此也優(yōu)先被執(zhí)行。

      完全掌握Mysql的explain

      如上圖所示,SQL查詢語句中,第一個(gè)執(zhí)行計(jì)劃的id為1,第二個(gè)執(zhí)行計(jì)劃的id為2,id為1的執(zhí)行計(jì)劃對應(yīng)的table為order,id為2的執(zhí)行計(jì)劃對應(yīng)的table是user_info,結(jié)合SQL語句,我們知道先執(zhí)行子查詢select id from user_info,而后再執(zhí)行關(guān)于表order的數(shù)據(jù)查詢。

      2、select_type

      select_type表示的執(zhí)行計(jì)劃的對應(yīng)的查詢是什么類型,常見的查詢類型主要包括普通查詢、聯(lián)合查詢以及子查詢等。SIMPLE(查詢語句為簡單的查詢不包含子查詢)、PRIMARY(當(dāng)查詢語句中包含子查詢的時(shí)候,對應(yīng)最外層的查詢類型)、UNION(union之后出現(xiàn)的select語句對應(yīng)的查詢類型會(huì)標(biāo)記此類型)、SUBQUERY(子查詢會(huì)被標(biāo)記為此類型)、DEPENDENT SUBQUERY(取決于外面的查詢 )

      完全掌握Mysql的explain

      3、table

      table代表表名稱,表示要查詢哪張表。當(dāng)然不一定是真實(shí)的表的名稱,也可能是表的別名或者臨時(shí)表。

      4、partitions

      partitions代表的是分區(qū)的概念,表示在進(jìn)行查詢時(shí),如果對應(yīng)的表都會(huì)死分區(qū)表,那么這里就會(huì)顯示具體的分區(qū)信息。

      5、type

      type是非常核心的屬性,需要重點(diǎn)掌握。它表示的是當(dāng)前通過什么樣的方式對數(shù)據(jù)庫表進(jìn)行分訪問。

      完全掌握Mysql的explain

      (1)system

      該表只有一行(相當(dāng)于系統(tǒng)表),數(shù)據(jù)量很小,查詢速度很快,system是const類型的特例。

      (2)const

      如果type是const,說明在進(jìn)行數(shù)據(jù)查詢的時(shí)候,命中了primary key或唯一索引,此類數(shù)據(jù)查詢速度非???。

      完全掌握Mysql的explain

      (3)eq_ref

      在進(jìn)行數(shù)據(jù)查詢的過程中,如果SQL語句中在表連接情況下可以基于聚簇索引或者非null值的唯一索引記性數(shù)據(jù)掃描,那么此時(shí)type對應(yīng)的值就會(huì)顯示為eq_ref。

      (4)ref

      數(shù)據(jù)查詢的時(shí)候如果命中的索引是二級索引不是唯一索引,測試查詢速度也會(huì)很快,但是type是ref。另外如果是多字段的聯(lián)合索引,那么根據(jù)最左匹配原則,從聯(lián)合索引的最左側(cè)開始連續(xù)多個(gè)列的字段進(jìn)行等值比較也是ref的類型。

      完全掌握Mysql的explain

      (5)ref_or_null

      這種連接類型類似于 ref,區(qū)別在于 MySQL會(huì)額外搜索包含NULL值的行。

      (7)unique_subquery

      在where條件中的關(guān)于in的子查詢條件集合

      (8)index_subquery

      區(qū)別于unique_subquery,用于非唯一索引,可以返回重復(fù)值。

      (9)range

      使用索引進(jìn)行行數(shù)據(jù)檢索,只對指定范圍內(nèi)的行數(shù)據(jù)進(jìn)行檢索。換句話說就是針對一個(gè)有索引的字段,在指定范圍中檢索數(shù)據(jù)。在where語句中使用 bettween…and、<、>、<=、in 等條件查詢 type 都是 range。

      完全掌握Mysql的explain

      (10)index

      Index 與ALL 其實(shí)都是讀全表,區(qū)別在于index是遍歷索引樹讀取,而ALL是從硬盤中讀取。

      (11)all

      遍歷全表進(jìn)行數(shù)據(jù)匹配,此時(shí)的數(shù)據(jù)查詢性能最差。

      6、possible_keys

      表示哪些索引可以被Mysql的優(yōu)化器進(jìn)行選擇,也就是索引候選者有哪些。

      7、key

      在possible_keys中實(shí)際選擇的索引

      8、key_len

      表示索引的長度,和實(shí)際的字段屬性以及是否為null都有關(guān)系。

      9、ref

      完全掌握Mysql的explain

      當(dāng)使用字段進(jìn)行常量等值查詢時(shí)ref此處為const,當(dāng)查詢條件中使用了表達(dá)式或者函數(shù)則ref顯示為func,則其他的顯示為null。

      10、rows

      rows列顯示MySQL認(rèn)為它執(zhí)行查詢時(shí)必須檢查的行數(shù)。行數(shù)越少,效率越高!

      11、filtered

      filtered 這個(gè)是一個(gè)百分比的值,表里符合條件的記錄數(shù)的百分比。簡單點(diǎn)說,這個(gè)字段表示存儲引擎返回的數(shù)據(jù)在經(jīng)過過濾后,剩下滿足條件的記錄數(shù)量的比例。

      12、extra

      在其他列不顯示額外信息在此列進(jìn)行展示。

      (1)Using index

      在進(jìn)行數(shù)據(jù)查詢的時(shí)候,數(shù)據(jù)庫使用了覆蓋索引,就是查詢的列被索引覆蓋,使用到覆蓋索引查詢速度會(huì)非常快。不是使用select * ,而是使用select phone_number,就會(huì)用到覆蓋索引。

      完全掌握Mysql的explain

      (2)Using where

      查詢時(shí)未找到可用的索引,進(jìn)而通過where條件過濾獲取所需數(shù)據(jù),但要注意的是并不是所有帶where語句的查詢都會(huì)顯示Using where。

      完全掌握Mysql的explain

      (3)Using temporary

      表示查詢后結(jié)果需要使用臨時(shí)表來存儲,一般在排序或者分組查詢時(shí)用到。

      (4)Using filesort

      此類型表示無法利用索引完成指定的排序操作,也就是ORDER BY的字段實(shí)際沒有索引,因此此類SQL是需要進(jìn)行優(yōu)化的。

      exlpain分析實(shí)戰(zhàn)

      上文中我們闡述了explain在分析SQL語句時(shí),可以通過12個(gè)屬性來分析SQL的大致執(zhí)行過程,并以此來判斷SQL存在的性能問題。那么接下來我們通過一個(gè)實(shí)際的例子,來具體看下如何結(jié)合explain來實(shí)現(xiàn)SQL的性能分析。

      其實(shí)所謂的Mysql性能問題,大部分都指的是平臺出現(xiàn)了慢查詢問題。慢查詢實(shí)際上是可以通過配置進(jìn)行記錄的,把執(zhí)行時(shí)間超過某個(gè)設(shè)定的閾值的sql都記錄下來,當(dāng)出現(xiàn)問題的時(shí)候可以通過記錄的慢查詢?nèi)罩具M(jìn)行問題的定位。但是有的時(shí)候,出現(xiàn)大量慢查詢會(huì)導(dǎo)致數(shù)據(jù)庫連接給占滿,導(dǎo)致整個(gè)平臺的出現(xiàn)異常。

      實(shí)際上我們在產(chǎn)品評價(jià)表product_evaluation中是建立了索引的,正常來說應(yīng)該是可以使用到對應(yīng)的索引字段進(jìn)行查詢的。但是實(shí)際上查詢耗時(shí)有幾十秒的時(shí)間,遠(yuǎn)遠(yuǎn)超過我們的預(yù)期。那我們猜測是不是由于某種原因?qū)е翸ysql優(yōu)化器沒有選擇對應(yīng)的索引進(jìn)行數(shù)據(jù)檢索,最后造成慢查詢的發(fā)生。到底執(zhí)行計(jì)劃是怎樣的,還是得借助于explain來看下。

      完全掌握Mysql的explain

      如上文所說,雖然explain有12個(gè)字段屬性幫助我們進(jìn)行執(zhí)行計(jì)劃的分析,但是實(shí)際上常用的核心字段也就幾個(gè)。我們可以看的出來在possible_key中實(shí)際上包含了我們設(shè)置的索引的,但是實(shí)際上Mysql卻選擇了PRIMARY作為其實(shí)際使用的。那么問題來了,為什么明明設(shè)置了索引,但是實(shí)際并沒有用上,唄Mysql吃了嗎?另外為什么之前的業(yè)務(wù)中沒有出現(xiàn)這個(gè)問題,而現(xiàn)在出現(xiàn)了?我們需要進(jìn)行進(jìn)一步的分析。

      我們所建立的idx_evaluation_type實(shí)際上是一個(gè)二級索引(葉子節(jié)點(diǎn)是主鍵id),對于數(shù)千萬一張的大表來說,實(shí)際上這個(gè)二級索引也是非常大的,而且這個(gè)字段本身的值就三個(gè),變化不大。因此Mysql的優(yōu)化器在分析這個(gè)SQL的時(shí)候發(fā)現(xiàn),如果按照SQL中的索引來獲取數(shù)據(jù)后再根據(jù)where條件進(jìn)行篩選,篩選后的數(shù)據(jù)還需要回表到聚簇索引中獲取實(shí)際的數(shù)據(jù)。

      假如通過二級索引篩選出來的數(shù)據(jù)有幾萬條,而后還需要進(jìn)行排序,這些操作都是基于臨時(shí)磁盤我恩建進(jìn)行的,Mysql判斷這種方式的性能可能會(huì)很差,因此優(yōu)化器放棄了原有的數(shù)據(jù)查詢方式,直接通過主鍵id對應(yīng)的聚簇索引來進(jìn)行數(shù)據(jù)的獲取,因?yàn)閕d本身就是有序的。

      完全掌握Mysql的explain

      那么知道了查詢慢的原因,我們應(yīng)該怎么進(jìn)行優(yōu)化呢?實(shí)際上可以在SQL語句中增加force idnex,強(qiáng)制Mysql使用我們設(shè)置的二級索引。

      SELECT * FROM product_evaluation force index(idx_product_id)WHERE product_id =1 and evaluation_type='GOOD'  ORDER BY id desc LIMIT 200

      總結(jié)

      通過上文對于explain使用的介紹,大家在遇到慢SQL問題的時(shí)候,可以先通過explain來進(jìn)行初步的分析,主要明確SQL在Mysql中實(shí)際的執(zhí)行過程是怎樣的,如果查詢字段沒有索引則增加索引,如果有索引就要分析為什么沒有用到索引。只要明確具體的執(zhí)行過程,我們才能確定具體的查詢優(yōu)化方案。

      推薦學(xué)習(xí):mysql視頻教程

      贊(0)
      分享到: 更多 (0)
      網(wǎng)站地圖   滬ICP備18035694號-2    滬公網(wǎng)安備31011702889846號