1. 程式人生 > 資料庫 >MySQL中有哪些情況下資料庫索引會失效詳析

MySQL中有哪些情況下資料庫索引會失效詳析

前言

要想分析MySQL查詢語句中的相關資訊,如是全表查詢還是部分查詢,就要用到explain.

索引的優點

  • 大大減少了伺服器需要掃描的資料量
  • 可以幫助伺服器避免排序或減少使用臨時表排序
  • 索引可以隨機I/O變為順序I/O

索引的缺點

  • 需要佔用磁碟空間,因此冗餘低效的索引將佔用大量的磁碟空間
  • 降低DML效能,對於資料的任意增刪改都需要調整對應的索引,甚至出現索引分裂
  • 索引會產生相應的碎片,產生維護開銷

一、explain

用法:explain +查詢語句。

id:查詢語句的序列號,上面圖片中只有一個select 語句,所以只會顯示一個序列號。如果有巢狀查詢,如下

select_type:表示查詢型別,有以下幾種

  simple:簡單的 select (沒有使用 union或子查詢)

  primary:最外層的 select。

  union:第二層,在select 之後使用了 union。

  dependent union:union 語句中的第二個select,依賴於外部子查詢

  subquery:子查詢中的第一個 select

  dependent subquery:子查詢中的第一個 subquery依賴於外部的子查詢

  derived:派生表 select(from子句中的子查詢)

table:查詢的表、結果集

type:全稱為"join type",意為連線型別。通俗的講就是mysql查詢引擎找到滿足SQL條件的資料的方式。其值為:

  • system:系統表,表中只有一行資料
  • const:讀常量,最多隻會有一條記錄匹配,由於是常量,實際上只須要讀一次。
  • eq_ref:最多隻會有一條匹配結果,一般是通過主鍵或唯一鍵索引來訪問。
  • ref:對於每個來自於前面的表的行組合,所有有匹配索引值的行將從這張表中讀取
  • fulltext:進行全文索引檢索。
  • ref_or_null:與ref的唯一區別就是在使用索引引用的查詢之外再增加一個空值的查詢。
  • index_merge:查詢中同時使用兩個(或更多)索引,然後對索引結果進行合併,再讀取表資料。
  • unique_subquery:子查詢中的返回結果欄位組合是主鍵或者唯一約束。
  • index_subquery:子查詢中的返回結果欄位組合是一個索引(或索引組合),但不是一個主鍵或唯一索引。
  • rang:索引範圍掃描。
  • index:全索引掃描。
  • all:全表掃描。

  效能從上到下依次降低。

possible_keys:可能用到的索引

key:使用的索引

ref:ref列顯示使用哪個列或常數與key一起從表中選擇行。

rows:顯示MySQL認為它執行查詢時必須檢查的行數。多行之間的資料相乘可以估算要處理的行數。

Extra:額外的資訊

  • Distinct:MySQL發現第1個匹配行後,停止為當前的行組合搜尋更多的行。
  • Not exists:MySQL能夠對查詢進行LEFT JOIN優化,發現1個匹配LEFT JOIN標準的行後,不再為前面的的行組合在該表內檢查更多的行。
  • range checked for each record (index map: #):MySQL沒有發現好的可以使用的索引,但發現如果來自前面的表的列值已知,可能部分索引可以使用。
  • Using filesort:MySQL需要額外的一次傳遞,以找出如何按排序順序檢索行。
  • Using index:從只使用索引樹中的資訊而不需要進一步搜尋讀取實際的行來檢索表中的列資訊。
  • Using temporary:為了解決查詢,MySQL需要建立一個臨時表來容納結果。
  • Using where:WHERE 子句用於限制哪一個行匹配下一個表或傳送到客戶。
  • Using sort_union(...),Using union(...),Using intersect(...):這些函式說明如何為index_merge聯接型別合併索引掃描。
  • Using index for group-by:類似於訪問表的Using index方式,Using index for group-by表示MySQL發現了一個索引,可以用來查 詢GROUP BY或DISTINCT查詢的所有列,而不要額外搜尋硬碟訪問實際的表。

二、資料庫不使用索引的情況

下面舉的例子中,GudiNo、StoreId列都有單獨的索引。

2.1、like查詢已 '%...'開頭,以'xxx%'結尾會繼續使用索引。

下圖中第一句使用的%,沒有使用索引,從rows為224147,使用索引rows為1。

    

2.2 where語句中使用 <>和 !=

2.3 where語句中使用 or,但是沒有把or中所有欄位加上索引。

這種情況,如果需要使用索引需要將or中所有的欄位都加上索引。

2.4 where語句中對欄位表示式操作

2.5 where語句中使用Not In


看了別人寫的文章,有說“應儘量避免在where 子句中對欄位進行null 值判斷,否則將導致引擎放棄使用索引而進行全表掃描”,實測沒有全表掃描。

"對於多列索引,不是使用的第一部分,則不會使用索引",實測即使多索引,沒有使用第一部分,也會命中索引,沒有全表掃描。

總結

以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對我們的支援。