mysql explain用法和結果的含義

explain顯示了mysql如何使用索引來處理select語句以及連接表。可以幫助選擇更好的索引和寫出更優化的查詢語句。

使用方法,在select語句前加上explain就可以了:

如:

  1. explain?select?surname,first_name?form?a,b?where?a.id=b.id?

EXPLAIN列的解釋:

table:顯示這一行的數據是關于哪張表的

type:這是重要的列,顯示連接使用了何種類型。從最好到最差的連接類型為const、eq_reg、ref、range、index和ALL

? ? ? ? ?type顯示的是訪問類型,是較為重要的一個指標,結果值從好到壞依次是:system > const > eq_ref > ref > fulltext > ref_or_null > index_merge > unique_subquery > index_subquery > range > index > ALL
? ? ? ? ?一般來說,得保證查詢至少達到range級別,最好能達到ref。

possible_keys:顯示可能應用在這張表中的索引。如果為空,沒有可能的索引。可以為相關的域從WHERE語句中選擇一個合適的語句

key: 實際使用的索引。如果為NULL,則沒有使用索引。很少的情況下,MYSQL會選擇優化不足的索引。這種情況下,可以在SELECT語句中使用USE INDEX(indexname)來強制使用一個索引或者用IGNORE INDEX(indexname)來強制MYSQL忽略索引

key_len:使用的索引的長度。在不損失精確性的情況下,長度越短越好

ref:顯示索引的哪一列被使用了,如果可能的話,是一個常數

rows:MYSQL認為必須檢查的用來返回請求數據的行數

Extra:關于MYSQL如何解析查詢的額外信息。將在表4.3中討論,但這里可以看到的壞的例子是Using temporary和Using filesort,意思MYSQL根本不能使用索引,結果是檢索會很慢

extra列返回的描述的意義

?Distinct:一旦MYSQL找到了與行相聯合匹配的行,就不再搜索了

?Not exists: MYSQL優化了LEFT JOIN,一旦它找到了匹配LEFT JOIN標準的行,就不再搜索了

?Range checked for each Record(index map:#):沒有找到理想的索引,因此對于從前面表中來的每一個行組合,MYSQL檢查使用哪個索引,并用它來從表中返回行。這是使用索引的最慢的連接之一

?Using filesort: 看到這個的時候,查詢就需要優化了。MYSQL需要進行額外的步驟來發現如何對返回的行排序。它根據連接類型以及存儲排序鍵值和匹配條件的全部行的行指針來排序全部行

?Using index: 列數據是從僅僅使用了索引中的信息而沒有讀取實際的行動的表返回的,這發生在對表的全部的請求列都是同一個索引的部分的時候

?Using temporary 看到這個的時候,查詢需要優化了。這里,MYSQL需要創建一個臨時表來存儲結果,這通常發生在對不同的列集進行ORDER BY上,而不是GROUP BY上

?Where used 使用了WHERE從句來限制哪些行將與下一張表匹配或者是返回給用戶。如果不想返回表中的全部行,并且連接類型ALL或index,這就會發生,或者是查詢有問題不同連接類型的解釋(按照效率高低的順序排序)

?system 表只有一行:system表。這是const連接類型的特殊情況

?const:表中的一個記錄的最大值能夠匹配這個查詢(索引可以是主鍵或惟一索引)。因為只有一行,這個值實際就是常數,因為MYSQL先讀這個值然后把它當做常數來對待

?eq_ref:在連接中,MYSQL在查詢時,從前面的表中,對每一個記錄的聯合都從表中讀取一個記錄,它在查詢使用了索引為主鍵或惟一鍵的全部時使用

?ref:這個連接類型只有在查詢使用了不是惟一或主鍵的鍵或者是這些類型的部分(比如,利用最左邊前綴)時發生。對于之前的表的每一個行聯合,全部記錄都將從表中讀出。這個類型嚴重依賴于根據索引匹配的記錄多少—越少越好

?range:這個連接類型使用索引返回一個范圍中的行,比如使用>或<查找東西時發生的情況

?index: 這個連接類型對前面的表中的每一個記錄聯合進行完全掃描(比ALL更好,因為索引一般小于表數據)

?ALL:這個連接類型對于前面的每一個記錄聯合進行完全掃描,這一般比較糟糕,應該盡量避免

??

先看一個例子:

  1. mysql>?explain?select?*?from?t_order;?
  2. +----+-------------+---------+------+---------------+------+---------+------+--------+-------+?
  3. |?id?|?select_type?|?table???|?type?|?possible_keys?|?key??|?key_len?|?ref??|?rows???|?Extra?|?
  4. +----+-------------+---------+------+---------------+------+---------+------+--------+-------+?
  5. |??1?|?SIMPLE??????|?t_order?|?ALL??|?NULL??????????|?NULL?|?NULL????|?NULL?|?100453?|???????|?
  6. +----+-------------+---------+------+---------------+------+---------+------+--------+-------+?
  7. 1?row?in?set?(0.03?sec)?

加上extended后之后:

  1. mysql>?explain?extended?select?*?from?t_order;?
  2. +----+-------------+---------+------+---------------+------+---------+------+--------+----------+-------+?
  3. |?id?|?select_type?|?table???|?type?|?possible_keys?|?key??|?key_len?|?ref??|?rows???|?filtered?|?Extra?|?
  4. +----+-------------+---------+------+---------------+------+---------+------+--------+----------+-------+?
  5. |??1?|?SIMPLE??????|?t_order?|?ALL??|?NULL??????????|?NULL?|?NULL????|?NULL?|?100453?|???100.00?|???????|?
  6. +----+-------------+---------+------+---------------+------+---------+------+--------+----------+-------+?
  7. 1?row?in?set,?1?warning?(0.00?sec)?

有必要解釋一下這個長長的表格里每一列的含義:

idSELECT識別符。這是SELECT的查詢序列號
select_type

SELECT類型,可以為以下任何一種:

  • SIMPLE:簡單SELECT(不使用UNION或子查詢)
  • PRIMARY:最外面的SELECT
  • UNION:UNION中的第二個或后面的SELECT語句
  • DEPENDENT UNION:UNION中的第二個或后面的SELECT語句,取決于外面的查詢
  • UNION RESULT:UNION 的結果
  • SUBQUERY:子查詢中的第一個SELECT
  • DEPENDENT SUBQUERY:子查詢中的第一個SELECT,取決于外面的查詢
  • DERIVED:導出表的SELECT(FROM子句的子查詢)
table

輸出的行所引用的表

type

聯接類型。下面給出各種聯接類型,按照從最佳類型到最壞類型進行排序:

  • system:表僅有一行(=系統表)。這是const聯接類型的一個特例。
  • const:表最多有一個匹配行,它將在查詢開始時被讀取。因為僅有一行,在這行的列值可被優化器剩余部分認為是常數。const表很快,因為它們只讀取一次!
  • eq_ref:對于每個來自于前面的表的行組合,從該表中讀取一行。這可能是最好的聯接類型,除了const類型。
  • ref:對于每個來自于前面的表的行組合,所有有匹配索引值的行將從這張表中讀取。
  • ref_or_null:該聯接類型如同ref,但是添加了MySQL可以專門搜索包含NULL值的行。
  • index_merge:該聯接類型表示使用了索引合并優化方法。
  • unique_subquery:該類型替換了下面形式的IN子查詢的ref: value IN (SELECT primary_key FROM single_table WHERE some_expr) unique_subquery是一個索引查找函數,可以完全替換子查詢,效率更高。
  • index_subquery:該聯接類型類似于unique_subquery。可以替換IN子查詢,但只適合下列形式的子查詢中的非唯一索引: value IN (SELECT key_column FROM single_table WHERE some_expr)
  • range:只檢索給定范圍的行,使用一個索引來選擇行。
  • index:該聯接類型與ALL相同,除了只有索引樹被掃描。這通常比ALL快,因為索引文件通常比數據文件小。
  • ALL:對于每個來自于先前的表的行組合,進行完整的表掃描。
possible_keys

指出MySQL能使用哪個索引在該表中找到行

key顯示MySQL實際決定使用的鍵(索引)。如果沒有選擇索引,鍵是NULL。
key_len顯示MySQL決定使用的鍵長度。如果鍵是NULL,則長度為NULL。
ref顯示使用哪個列或常數與key一起從表中選擇行。
rows顯示MySQL認為它執行查詢時必須檢查的行數。多行之間的數據相乘可以估算要處理的行數。
filtered顯示了通過條件過濾出的行數的百分比估計值。
Extra

該列包含MySQL解決查詢的詳細信息

  • 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查詢的所有列,而不要額外搜索硬盤訪問實際的表。

?

一.select_type的說明

1.UNION:

當通過union來連接多個查詢結果時,第二個之后的select其select_type為UNION。

  1. mysql>?explain?select?*?from?t_order?where?order_id=100?union?select?*?from?t_order?where?order_id=200;?
  2. +----+--------------+------------+-------+---------------+---------+---------+-------+------+-------+?
  3. |?id?|?select_type??|?table??????|?type??|?possible_keys?|?key?????|?key_len?|?ref???|?rows?|?Extra?|?
  4. +----+--------------+------------+-------+---------------+---------+---------+-------+------+-------+?
  5. |??1?|?PRIMARY??????|?t_order????|?const?|?PRIMARY???????|?PRIMARY?|?4???????|?const?|????1?|???????|?
  6. |??2?|?UNION????????|?t_order????|?const?|?PRIMARY???????|?PRIMARY?|?4???????|?const?|????1?|???????|?
  7. |?NULL?|?UNION?RESULT?|?<union1,2>?|?ALL???|?NULL??????????|?NULL????|?NULL????|?NULL??|?NULL?|???????|?
  8. +----+--------------+------------+-------+---------------+---------+---------+-------+------+-------+?
  9. 3?rows?in?set?(0.34?sec)?

2.DEPENDENT UNION與DEPENDENT SUBQUERY:

當union作為子查詢時,其中第二個union的select_type就是DEPENDENT UNION。
第一個子查詢的select_type則是DEPENDENT SUBQUERY。

  1. mysql>?explain?select?*?from?t_order?where?order_id?in?(select?order_id?from?t_order?where?order_id=100?union?select?order_id?from?t_order?where?order_id=200);?
  2. +----+--------------------+------------+-------+---------------+---------+---------+-------+--------+-------------+?
  3. |?id?|?select_type????????|?table??????|?type??|?possible_keys?|?key?????|?key_len?|?ref???|?rows???|?Extra???????|?
  4. +----+--------------------+------------+-------+---------------+---------+---------+-------+--------+-------------+?
  5. |??1?|?PRIMARY????????????|?t_order????|?ALL???|?NULL??????????|?NULL????|?NULL????|?NULL??|?100453?|?Using?where?|?
  6. |??2?|?DEPENDENT?SUBQUERY?|?t_order????|?const?|?PRIMARY???????|?PRIMARY?|?4???????|?const?|??????1?|?Using?index?|?
  7. |??3?|?DEPENDENT?UNION????|?t_order????|?const?|?PRIMARY???????|?PRIMARY?|?4???????|?const?|??????1?|?Using?index?|?
  8. |?NULL?|?UNION?RESULT???????|?<union2,3>?|?ALL???|?NULL??????????|?NULL????|?NULL????|?NULL??|???NULL?|?????????????|?
  9. +----+--------------------+------------+-------+---------------+---------+---------+-------+--------+-------------+?
  10. 4?rows?in?set?(0.03?sec)?

3.SUBQUERY:

子查詢中的第一個select其select_type為SUBQUERY。

  1. mysql>?explain?select?*?from?t_order?where?order_id=(select?order_id?from?t_order?where?order_id=100);?
  2. +----+-------------+---------+-------+---------------+---------+---------+-------+------+-------------+?
  3. |?id?|?select_type?|?table???|?type??|?possible_keys?|?key?????|?key_len?|?ref???|?rows?|?Extra???????|?
  4. +----+-------------+---------+-------+---------------+---------+---------+-------+------+-------------+?
  5. |??1?|?PRIMARY?????|?t_order?|?const?|?PRIMARY???????|?PRIMARY?|?4???????|?const?|????1?|?????????????|?
  6. |??2?|?SUBQUERY????|?t_order?|?const?|?PRIMARY???????|?PRIMARY?|?4???????|???????|????1?|?Using?index?|?
  7. +----+-------------+---------+-------+---------------+---------+---------+-------+------+-------------+?
  8. 2?rows?in?set?(0.03?sec)?

4.DERIVED:

當子查詢是from子句時,其select_type為DERIVED。

  1. mysql>?explain?select?*?from?(select?order_id?from?t_order?where?order_id=100)?a;?
  2. +----+-------------+------------+--------+---------------+---------+---------+------+------+-------------+?
  3. |?id?|?select_type?|?table??????|?type???|?possible_keys?|?key?????|?key_len?|?ref??|?rows?|?Extra???????|?
  4. +----+-------------+------------+--------+---------------+---------+---------+------+------+-------------+?
  5. |??1?|?PRIMARY?????|?<derived2>?|?system?|?NULL??????????|?NULL????|?NULL????|?NULL?|????1?|?????????????|?
  6. |??2?|?DERIVED?????|?t_order????|?const??|?PRIMARY???????|?PRIMARY?|?4???????|??????|????1?|?Using?index?|?
  7. +----+-------------+------------+--------+---------------+---------+---------+------+------+-------------+?
  8. 2?rows?in?set?(0.03?sec)?

二.type的說明

1.system,const

見上面4.DERIVED的例子。其中第一行的type就是為system,第二行是const,這兩種聯接類型是最快的。

2.eq_ref

在t_order表中的order_id是主鍵,t_order_ext表中的order_id也是主鍵,該表可以認為是訂單表的補充信息表,他們的關系是1對1,在下面的例子中可以看到b表的連接類型是eq_ref,這是極快的聯接類型。

  1. mysql>?explain?select?*?from?t_order?a,t_order_ext?b?where?a.order_id=b.order_id;?
  2. +----+-------------+-------+--------+---------------+---------+---------+-----------------+------+-------------+?
  3. |?id?|?select_type?|?table?|?type???|?possible_keys?|?key?????|?key_len?|?ref?????????????|?rows?|?Extra???????|?
  4. +----+-------------+-------+--------+---------------+---------+---------+-----------------+------+-------------+?
  5. |??1?|?SIMPLE??????|?b?????|?ALL????|?order_id??????|?NULL????|?NULL????|?NULL????????????|????1?|?????????????|?
  6. |??1?|?SIMPLE??????|?a?????|?eq_ref?|?PRIMARY???????|?PRIMARY?|?4???????|?test.b.order_id?|????1?|?Using?where?|?
  7. +----+-------------+-------+--------+---------------+---------+---------+-----------------+------+-------------+?
  8. 2?rows?in?set?(0.00?sec)?

3.ref

下面的例子在上面的例子上略作了修改,加上了條件。此時b表的聯接類型變成了ref。因為所有與a表中order_id=100的匹配記錄都將會從b表獲取。這是比較常見的聯接類型。

  1. mysql>?explain?select?*?from?t_order?a,t_order_ext?b?where?a.order_id=b.order_id?and?a.order_id=100;?
  2. +----+-------------+-------+-------+---------------+----------+---------+-------+------+-------+?
  3. |?id?|?select_type?|?table?|?type??|?possible_keys?|?key??????|?key_len?|?ref???|?rows?|?Extra?|?
  4. +----+-------------+-------+-------+---------------+----------+---------+-------+------+-------+?
  5. |??1?|?SIMPLE??????|?a?????|?const?|?PRIMARY???????|?PRIMARY??|?4???????|?const?|????1?|???????|?
  6. |??1?|?SIMPLE??????|?b?????|?ref???|?order_id??????|?order_id?|?4???????|?const?|????1?|???????|?
  7. +----+-------------+-------+-------+---------------+----------+---------+-------+------+-------+?
  8. 2?rows?in?set?(0.00?sec)?

4.ref_or_null

user_id字段是一個可以為空的字段,并對該字段創建了一個索引。在下面的查詢中可以看到聯接類型為ref_or_null,這是mysql為含有null的字段專門做的處理。在我們的表設計中應當盡量避免索引字段為NULL,因為這會額外的耗費mysql的處理時間來做優化。

  1. mysql>?explain?select?*?from?t_order?where?user_id=100?or?user_id?is?null;?
  2. +----+-------------+---------+-------------+---------------+---------+---------+-------+-------+-------------+?
  3. |?id?|?select_type?|?table???|?type????????|?possible_keys?|?key?????|?key_len?|?ref???|?rows??|?Extra???????|?
  4. +----+-------------+---------+-------------+---------------+---------+---------+-------+-------+-------------+?
  5. |??1?|?SIMPLE??????|?t_order?|?ref_or_null?|?user_id???????|?user_id?|?5???????|?const?|?50325?|?Using?where?|?
  6. +----+-------------+---------+-------------+---------------+---------+---------+-------+-------+-------------+?
  7. 1?row?in?set?(0.00?sec)?

5.index_merge

經常出現在使用一張表中的多個索引時。mysql會將多個索引合并在一起,如下例:

  1. mysql>?explain?select?*?from?t_order?where?order_id=100?or?user_id=10;?
  2. +----+-------------+---------+-------------+-----------------+-----------------+---------+------+------+-------------------------------------------+?
  3. |?id?|?select_type?|?table???|?type????????|?possible_keys???|?key?????????????|?key_len?|?ref??|?rows?|?Extra?????????????????????????????????????|?
  4. +----+-------------+---------+-------------+-----------------+-----------------+---------+------+------+-------------------------------------------+?
  5. |??1?|?SIMPLE??????|?t_order?|?index_merge?|?PRIMARY,user_id?|?PRIMARY,user_id?|?4,5?????|?NULL?|????2?|?Using?union(PRIMARY,user_id);?Using?where?|?
  6. +----+-------------+---------+-------------+-----------------+-----------------+---------+------+------+-------------------------------------------+?
  7. 1?row?in?set?(0.09?sec)?

6.unique_subquery

該聯接類型用于替換value IN (SELECT primary_key FROM single_table WHERE some_expr)這樣的子查詢的ref。注意ref列,其中第二行顯示的是func,表明unique_subquery是一個函數,而不是一個普通的ref。

  1. mysql>?explain?select?*?from?t_order?where?order_id?in?(select?order_id?from?t_order?where?user_id=10);?
  2. +----+--------------------+---------+-----------------+-----------------+---------+---------+------+--------+-------------+?
  3. |?id?|?select_type????????|?table???|?type????????????|?possible_keys???|?key?????|?key_len?|?ref??|?rows???|?Extra???????|?
  4. +----+--------------------+---------+-----------------+-----------------+---------+---------+------+--------+-------------+?
  5. |??1?|?PRIMARY????????????|?t_order?|?ALL?????????????|?NULL????????????|?NULL????|?NULL????|?NULL?|?100649?|?Using?where?|?
  6. |??2?|?DEPENDENT?SUBQUERY?|?t_order?|?unique_subquery?|?PRIMARY,user_id?|?PRIMARY?|?4???????|?func?|??????1?|?Using?where?|?
  7. +----+--------------------+---------+-----------------+-----------------+---------+---------+------+--------+-------------+?
  8. 2?rows?in?set?(0.00?sec)?

7.index_subquery

該聯接類型與上面的太像了,唯一的差別就是子查詢查的不是主鍵而是非唯一索引。

  1. mysql>?explain?select?*?from?t_order?where?user_id?in?(select?user_id?from?t_order?where?order_id>10);?
  2. +----+--------------------+---------+----------------+-----------------+---------+---------+------+--------+--------------------------+?
  3. |?id?|?select_type????????|?table???|?type???????????|?possible_keys???|?key?????|?key_len?|?ref??|?rows???|?Extra????????????????????|?
  4. +----+--------------------+---------+----------------+-----------------+---------+---------+------+--------+--------------------------+?
  5. |??1?|?PRIMARY????????????|?t_order?|?ALL????????????|?NULL????????????|?NULL????|?NULL????|?NULL?|?100649?|?Using?where??????????????|?
  6. |??2?|?DEPENDENT?SUBQUERY?|?t_order?|?index_subquery?|?PRIMARY,user_id?|?user_id?|?5???????|?func?|??50324?|?Using?index;?Using?where?|?
  7. +----+--------------------+---------+----------------+-----------------+---------+---------+------+--------+--------------------------+?
  8. 2?rows?in?set?(0.00?sec)?

8.range

按指定的范圍進行檢索,很常見。

  1. mysql>?explain?select?*?from?t_order?where?user_id?in?(100,200,300);?
  2. +----+-------------+---------+-------+---------------+---------+---------+------+------+-------------+?
  3. |?id?|?select_type?|?table???|?type??|?possible_keys?|?key?????|?key_len?|?ref??|?rows?|?Extra???????|?
  4. +----+-------------+---------+-------+---------------+---------+---------+------+------+-------------+?
  5. |??1?|?SIMPLE??????|?t_order?|?range?|?user_id???????|?user_id?|?5???????|?NULL?|????3?|?Using?where?|?
  6. +----+-------------+---------+-------+---------------+---------+---------+------+------+-------------+?
  7. 1?row?in?set?(0.00?sec)?

9.index

在進行統計時非常常見,此聯接類型實際上會掃描索引樹,僅比ALL快些。

  1. mysql>?explain?select?count(*)?from?t_order;?
  2. +----+-------------+---------+-------+---------------+---------+---------+------+--------+-------------+?
  3. |?id?|?select_type?|?table???|?type??|?possible_keys?|?key?????|?key_len?|?ref??|?rows???|?Extra???????|?
  4. +----+-------------+---------+-------+---------------+---------+---------+------+--------+-------------+?
  5. |??1?|?SIMPLE??????|?t_order?|?index?|?NULL??????????|?user_id?|?5???????|?NULL?|?100649?|?Using?index?|?
  6. +----+-------------+---------+-------+---------------+---------+---------+------+--------+-------------+?
  7. 1?row?in?set?(0.00?sec)?

10.ALL

完整的掃描全表,最慢的聯接類型,盡可能的避免。

  1. mysql>?explain?select?*?from?t_order;?
  2. +----+-------------+---------+------+---------------+------+---------+------+--------+-------+?
  3. |?id?|?select_type?|?table???|?type?|?possible_keys?|?key??|?key_len?|?ref??|?rows???|?Extra?|?
  4. +----+-------------+---------+------+---------------+------+---------+------+--------+-------+?
  5. |??1?|?SIMPLE??????|?t_order?|?ALL??|?NULL??????????|?NULL?|?NULL????|?NULL?|?100649?|???????|?
  6. +----+-------------+---------+------+---------------+------+---------+------+--------+-------+?
  7. 1?row?in?set?(0.00?sec)?

三.extra的說明

1.Distinct

MySQL發現第1個匹配行后,停止為當前的行組合搜索更多的行。對于此項沒有找到合適的例子,求指點。

2.Not exists

因為b表中的order_id是主鍵,不可能為NULL,所以mysql在用a表的order_id掃描t_order表,并查找b表的行時,如果在b表發現一個匹配的行就不再繼續掃描b了,因為b表中的order_id字段不可能為NULL。這樣避免了對b表的多次掃描。

  1. mysql>?explain?select?count(1)?from?t_order?a?left?join?t_order_ext?b?on?a.order_id=b.order_id?where?b.order_id?is?null;??
  2. +----+-------------+-------+-------+---------------+--------------+---------+-----------------+--------+--------------------------------------+?
  3. |?id?|?select_type?|?table?|?type??|?possible_keys?|?key??????????|?key_len?|?ref?????????????|?rows???|?Extra????????????????????????????????|?
  4. +----+-------------+-------+-------+---------------+--------------+---------+-----------------+--------+--------------------------------------+?
  5. |??1?|?SIMPLE??????|?a?????|?index?|?NULL??????????|?express_type?|?1???????|?NULL????????????|?100395?|?Using?index??????????????????????????|?
  6. |??1?|?SIMPLE??????|?b?????|?ref???|?order_id??????|?order_id?????|?4???????|?test.a.order_id?|??????1?|?Using?where;?Using?index;?Not?exists?|?
  7. +----+-------------+-------+-------+---------------+--------------+---------+-----------------+--------+--------------------------------------+?
  8. 2?rows?in?set?(0.01?sec)?

3.Range checked for each record

這種情況是mysql沒有發現好的索引可用,速度比沒有索引要快得多。

  1. mysql>?explain?select?*?from?t_order?t,?t_order_ext?s?where?s.order_id>=t.order_id?and?s.order_id<=t.order_id?and?t.express_type>5;?
  2. +----+-------------+-------+-------+----------------------+--------------+---------+------+------+------------------------------------------------+?
  3. |?id?|?select_type?|?table?|?type??|?possible_keys????????|?key??????????|?key_len?|?ref??|?rows?|?Extra??????????????????????????????????????????|?
  4. +----+-------------+-------+-------+----------------------+--------------+---------+------+------+------------------------------------------------+?
  5. |??1?|?SIMPLE??????|?t?????|?range?|?PRIMARY,express_type?|?express_type?|?1???????|?NULL?|????1?|?Using?where????????????????????????????????????|?
  6. |??1?|?SIMPLE??????|?s?????|?ALL???|?order_id?????????????|?NULL?????????|?NULL????|?NULL?|????1?|?Range?checked?for?each?record?(index?map:?0x1)?|?
  7. +----+-------------+-------+-------+----------------------+--------------+---------+------+------+------------------------------------------------+?
  8. 2?rows?in?set?(0.00?sec)

4.Using filesort

在有排序子句的情況下很常見的一種情況。此時mysql會根據聯接類型瀏覽所有符合條件的記錄,并保存排序關鍵字和行指針,然后排序關鍵字并按順序檢索行。

  1. mysql>?explain?select?*?from?t_order?order?by?express_type;?
  2. +----+-------------+---------+------+---------------+------+---------+------+--------+----------------+?
  3. |?id?|?select_type?|?table???|?type?|?possible_keys?|?key??|?key_len?|?ref??|?rows???|?Extra??????????|?
  4. +----+-------------+---------+------+---------------+------+---------+------+--------+----------------+?
  5. |??1?|?SIMPLE??????|?t_order?|?ALL??|?NULL??????????|?NULL?|?NULL????|?NULL?|?100395?|?Using?filesort?|?
  6. +----+-------------+---------+------+---------------+------+---------+------+--------+----------------+?
  7. 1?row?in?set?(0.00?sec)?

5.Using index

這是性能很高的一種情況。當查詢所需的數據可以直接從索引樹中檢索到時,就會出現。上面的例子中有很多這樣的例子,不再多舉例了。

6.Using temporary

發生這種情況一般都是需要進行優化的。mysql需要創建一張臨時表用來處理此類查詢。

  1. mysql>?explain?select?*?from?t_order?a?left?join?t_order_ext?b?on?a.order_id=b.order_id?group?by?b.order_id;?
  2. +----+-------------+-------+------+---------------+----------+---------+-----------------+--------+---------------------------------+?
  3. |?id?|?select_type?|?table?|?type?|?possible_keys?|?key??????|?key_len?|?ref?????????????|?rows???|?Extra???????????????????????????|?
  4. +----+-------------+-------+------+---------------+----------+---------+-----------------+--------+---------------------------------+?
  5. |??1?|?SIMPLE??????|?a?????|?ALL??|?NULL??????????|?NULL?????|?NULL????|?NULL????????????|?100395?|?Using?temporary;?Using?filesort?|?
  6. |??1?|?SIMPLE??????|?b?????|?ref??|?order_id??????|?order_id?|?4???????|?test.a.order_id?|??????1?|?????????????????????????????????|?
  7. +----+-------------+-------+------+---------------+----------+---------+-----------------+--------+---------------------------------+?
  8. 2?rows?in?set?(0.00?sec)?

7.Using where

當有where子句時,extra都會有說明。

8.Using sort_union(...)/Using union(...)/Using intersect(...)

下面的例子中user_id是一個檢索范圍,此時mysql會使用sort_union函數來進行索引的合并。而當user_id是一個固定值時,請參看上面type說明5.index_merge的例子,此時會使用union函數進行索引合并。

  1. mysql>?explain?select?*?from?t_order?where?order_id=100?or?user_id>10;?
  2. +----+-------------+---------+-------------+-----------------+-----------------+---------+------+------+------------------------------------------------+?
  3. |?id?|?select_type?|?table???|?type????????|?possible_keys???|?key?????????????|?key_len?|?ref??|?rows?|?Extra??????????????????????????????????????????|?
  4. +----+-------------+---------+-------------+-----------------+-----------------+---------+------+------+------------------------------------------------+?
  5. |??1?|?SIMPLE??????|?t_order?|?index_merge?|?PRIMARY,user_id?|?user_id,PRIMARY?|?5,4?????|?NULL?|????2?|?Using?sort_union(user_id,PRIMARY);?Using?where?|?
  6. +----+-------------+---------+-------------+-----------------+-----------------+---------+------+------+------------------------------------------------+?
  7. 1?row?in?set?(0.00?sec)?

對于Using intersect的例子可以參看下例,user_id與express_type發生了索引交叉合并。

  1. mysql>?explain?select?*?from?t_order?where?express_type=1?and?user_id=100;?
  2. +----+-------------+---------+-------------+----------------------+----------------------+---------+------+------+----------------------------------------------------+?
  3. |?id?|?select_type?|?table???|?type????????|?possible_keys????????|?key??????????????????|?key_len?|?ref??|?rows?|?Extra??????????????????????????????????????????????|?
  4. +----+-------------+---------+-------------+----------------------+----------------------+---------+------+------+----------------------------------------------------+?
  5. |??1?|?SIMPLE??????|?t_order?|?index_merge?|?user_id,express_type?|?user_id,express_type?|?5,1?????|?NULL?|????1?|?Using?intersect(user_id,express_type);?Using?where?|?
  6. +----+-------------+---------+-------------+----------------------+----------------------+---------+------+------+----------------------------------------------------+?
  7. 1?row?in?set?(0.00?sec)?

9.Using index for group-by

表明可以在索引中找到分組所需的所有數據,不需要查詢實際的表。

  1. mysql>?explain?select?user_id?from?t_order?group?by?user_id;?
  2. +----+-------------+---------+-------+---------------+---------+---------+------+------+--------------------------+?
  3. |?id?|?select_type?|?table???|?type??|?possible_keys?|?key?????|?key_len?|?ref??|?rows?|?Extra????????????????????|?
  4. +----+-------------+---------+-------+---------------+---------+---------+------+------+--------------------------+?
  5. |??1?|?SIMPLE??????|?t_order?|?range?|?NULL??????????|?user_id?|?5???????|?NULL?|????3?|?Using?index?for?group-by?|?
  6. +----+-------------+---------+-------+---------------+---------+---------+------+------+--------------------------+?
  7. 1?row?in?set?(0.00?sec)?

除了上面的三個說明,還需要注意rows的數值,多行之間的數值是乘積的關系,可以估算大概要處理的行數,如果乘積很大,那就很有優化的必要了。

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/386978.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/386978.shtml
英文地址,請注明出處:http://en.pswp.cn/news/386978.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

日志模塊logging用法

一、常用日志記錄場景及最佳解決方案&#xff1a; 日志記錄方式 最佳記錄日志方案 普通情況下&#xff0c;在控制臺顯示輸出 print() 報告正常程序操作過程中發生的事件 logging.info()(或者更詳細的logging.debug()) 發出有關特定事件的警告 warnings.warn()或者loggin…

MySQL 億級數據需求的優化思路(一),交易流水記錄的查詢

對MySQL的性能和億級數據的處理方法思考&#xff0c;以及分庫分表到底該如何做&#xff0c;在什么場景比較合適&#xff1f; 比如銀行交易流水記錄的查詢 限鹽少許&#xff0c;上實際實驗過程&#xff0c;以下是在實驗的過程中做一些操作&#xff0c;以及踩過的一些坑&#…

RUNOOB python練習題2

用來練手的python 練習題&#xff0c;原鏈接 : python練習實例2 題干 : 企業發放的獎金根據利潤提成。利潤(I)低于或等于10萬元時&#xff0c;獎金可提10%&#xff1b;利潤高于10萬元&#xff0c;低于20萬元時&#xff0c;低于10萬元的部分按10%提成&#xff0c;高于10萬元的…

dubbo負載均衡策略和集群容錯策略

dubbo負載均衡策略 random loadbalance 默認情況下&#xff0c;dubbo是random load balance隨機調用實現負載均衡&#xff0c;可以對provider不同實例設置不同的權重&#xff0c;會按照權重來負載均衡&#xff0c;權重越大分配流量越高&#xff0c;一般就用這個默認的就可以了。…

MySQL 億級數據需求的優化思路(二),100億數據,1萬字段屬性的秒級檢索

最近在研究億級數據的時候&#xff0c;無意中看到了一個關于寫58同城的文章 https://blog.csdn.net/admin1973/article/details/55251499?fromtimeline 其實上面講的versionext的方式以及壓縮json的思路&#xff0c;對于我來講都可以看得懂&#xff0c;想得通&#xff0c;其…

RUNOOB python練習題3

用來練手的python 練習題&#xff0c;原鏈接 : python練習實例3 拿到題目就寫了如下代碼&#xff0c;思路是因為使用**0.5進行開平方操作時&#xff0c;python會將數據類型自動轉換為float單精度浮點型。這里利用提取其整數部分&#xff0c;來判斷這個數是否是完全平方數。 z…

使用git將項目上傳到github(最簡單方法)

使用git將項目上傳到github&#xff08;最簡單方法&#xff09; 首先你需要一個github賬號&#xff0c;所有還沒有的話先去注冊吧&#xff01; https://github.com/ 我們使用git需要先安裝git工具&#xff0c;這里給出下載地址&#xff0c;下載后一路直接安裝即可&#xff1…

數據庫 概念詳解

數據庫 概念詳解 一、MySQL MySQL 事務 MySQL 鎖 MySQL 二、Redis 三、MongoDB 四、Memcached 轉載于:https://www.cnblogs.com/guozepingboke/p/10743648.html

RUNOOB python練習題4

用來練手的python習題其四&#xff0c; 原題鏈接: python練習實例4 題干: 輸入某年某月某日&#xff0c;判斷這一天是這一年的第幾天&#xff1f; 這個題目比較簡單&#xff0c;只需要注意閏年和非閏年的區別就可以了。我這里使用numpy矩陣存儲每個月的天數&#xff0c;之后用…

GitHub入門:如何上傳與下載工程?

由于經常要在家寫代碼&#xff0c;所以需要有個能夠方便訪問代碼管理工具。最近嘗試了一下GitHub。經過了一翻糾結之后&#xff0c;基本上掌握了他的使用方式。 要使用GitHub需要首先在其網站上進行注冊。其官方網站是https://github.com/。注冊的流程在這里就不多少了&#x…

如何解決PIP命令不可用

今天想用PIP裝一個python包&#xff0c;發現PIP報錯&#xff0c;不是內部或外部命令。。。 遇事百度&#xff0c;有兩種說法&#xff0c;一&#xff0c;沒安裝包&#xff0c;不管那么多命令執行了再說 在命令行輸入&#xff1a;python -m ensurepip 將pip.exe文件下載下來 再pi…

RUNOOB python練習題5

用來練手的python 練習題其五&#xff0c;原鏈接 : python練習實例5 題干 : 輸入三個整數x,y,z&#xff0c;請把這三個數由小到大輸出。 又是非常簡單的排序算法&#xff0c;只要使用numpy矩陣的排序方法或者使用python list的排序算法就可以輕松解決。 源代碼如下 : import …

初步使用github,并上傳下載文件

使用GitHub需要先注冊GitHub的賬號,登陸進去 然后開始創建項目 start a project 創建完成,開始生成公私鑰,可以不必每次都要輸密碼 ssh-keygen -t rsa -C "mghxy123163.com" //填寫email地址&#xff0c;然后一直“回車”ok 然后把公鑰導入GitHub中的key里面去,也…

NOIP2000提高組復賽C 單詞接龍

題目鏈接&#xff1a;https://ac.nowcoder.com/acm/contest/248/C 題目大意&#xff1a; 略 分析&#xff1a; 注意點&#xff1a;1.前綴和后綴的公共部分應該選最短的。2.如果兩個字符串前綴和后綴的公共部分恰好是其中一個字符串&#xff0c;那么這兩個字符串不能合并。 代碼…

右鍵Git Bash Here不見了怎么辦,手把手教你還原!

第一步&#xff0c;window R&#xff0c;輸入regedit回車進入注冊表 依次進入HKEY_CLASSES_ROOT —-》 Directory —-》Background —-》 shell 右鍵點擊shell&#xff0c;選擇新建&#xff0c;然后選擇項&#xff0c;命名為 Git Bash Here&#xff0c;成功后進入桌面右鍵發現…

RUNOOB python練習題6 斐波那契數列

用來練手的python 練習題其六&#xff0c;原鏈接 : python練習實例6 題干 : 斐波那契數列 斐波那契數列可以說是很好的遞歸理解工具了&#xff0c;這里就用遞歸實現一下斐波那契數列。 源代碼如下: # 返回fibonacci數列中某一項的數值 def Fibonacci(n):if n 1:return 1eli…

linux 單用戶密碼修改

1.啟動系統&#xff0c;并在GRUB2啟動屏顯時&#xff0c;按下e鍵進入編輯模式。 2.在linux16/inux/linuxef所在參數行ro更改為init/sysroot/bin/sh 3.按Crlx啟動到shell. 4.掛載文件系統為可寫模式: mount -o remount &#xff0c;rw /sysroot 5換根chroot /sysroot 6.運行pass…

github windows客戶端

方法/步驟 1 1. 首先到官網下載Github客戶端 2 2. 點擊上圖紅框的按鈕開始下載客戶端。 3 3. 雙擊下載好的客戶端&#xff0c;開始安裝。 4 雙擊之后出現一個框 5 之后等待一段時間&#xff0c;出現一個在線下載界面 6 4. 在線下載完成之后開始進行安裝。安裝完成之后…

賦值語句 變量的地址相關 : RUNOOB python練習題7

用來練手的python 練習題&#xff0c;原鏈接 : python練習實例7 練習實例7非常的簡單也有意思。題干 : 將一個列表的數據復制到另一個列表中。 完成這個操作的代碼非常簡單&#xff0c;即使是我這樣的初學者應該也是一語道破&#xff0c;賦值語句嘛。但這里我們就列舉出幾種不…

Web標準的概念及組成

一周更新兩個或三個關于web前端的知識點&#xff0c;歡迎感興趣的小伙伴們一起學習討論1、WEB標準是網頁制作的標準&#xff0c;它不是一個標準&#xff0c;它是根據網頁的不同組成部分生成的一系列標準。這些標準大部分由W3C起草發布&#xff0c;也有部分標準由ECMA起草發布。…