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

簡單談?wù)凪ySQL的loose index scan

 更新時(shí)間:2015年12月11日 12:01:52   投稿:hebedich  
各種資料關(guān)于loose index scan的解釋很拗口,其實(shí)等同于oracle的index skip scan。今天我們就來詳細(xì)看看loose index scan的使用

眾所周知,InnoDB采用IOT(index organization table)即所謂的索引組織表,而葉子節(jié)點(diǎn)也就存放了所有的數(shù)據(jù),這就意味著,數(shù)據(jù)總是按照某種順序存儲(chǔ)的。所以問題來了,如果是這樣一個(gè)語句,執(zhí)行起來應(yīng)該是怎么樣的呢?語句如下:

select count(distinct a) from table1;

     列a上有一個(gè)索引,那么按照簡單的想法來講,如何掃描呢?很簡單,一條一條的掃描,這樣一來,其實(shí)做了一次索引全掃描,效率很差。這種掃描方式會(huì)掃描到很多很多的重復(fù)的索引,這樣說的話優(yōu)化的辦法也是很容易想到的:跳過重復(fù)的索引就可以了。于是網(wǎng)上能搜到這樣的一個(gè)優(yōu)化的辦法:

select count(*) from (select distinct a from table1) t;

    從已經(jīng)搜索到的資料看,這樣的執(zhí)行計(jì)劃中的extra就從using index變成了using index for group-by。

    但是,但是,但是,好在我們現(xiàn)在已經(jīng)沒有使用5.1的版本了,大家基本上都是5.5以上了,這些現(xiàn)代版本,已經(jīng)實(shí)現(xiàn)了loose index scan:

     很好很好,就不需要再用這種奇技淫巧去優(yōu)化SQL了。

     文檔里關(guān)于group by這里寫的有點(diǎn)意思,說是最大眾化的辦法就是進(jìn)行全表掃描并且創(chuàng)建一個(gè)臨時(shí)表,這樣執(zhí)行計(jì)劃就會(huì)難看的要命了,肯定有ALL和using temporary table了。

5.0之后group by在特定條件下可能使用到loose index scan,

CREATE TABLE log_table (
id INT NOT NULL PRIMARY KEY,
log_machine VARCHAR(20) NOT NULL,
log_time DATETIME NOT NULL
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
CREATE INDEX ix_log_machine_time ON log_table (log_machine, log_time);

1

SELECT MAX(log_time) FROM log_table;
SELECT MAX(log_time) FROM log_table WHERE log_machine IN ('Machine 1');

這兩條sql都只需一次index seek便可返回,源于索引的有序排序,優(yōu)化器意識(shí)到min/max位于最左/右塊,從而避免范圍掃描;
extra顯示Select tables optimized away ;
2

復(fù)制代碼 代碼如下:
SELECT MAX(log_time) FROM log_table WHERE log_machine IN (‘Machine 1','Machine 2','Machine 3','Machine 4');

執(zhí)行計(jì)劃type 為range(extra顯示using where; using index),即執(zhí)行索引范圍掃描,先讀取所有滿足log_machine約束的記錄,然后對其遍歷找出max value;
改進(jìn)

復(fù)制代碼 代碼如下:
SELECT MAX(log_time) FROM log_table WHERE log_machine IN (‘Machine 1','Machine 2','Machine 3','Machine 4')  group by log_machine order by 1 desc limit 1;

這滿足group by選擇loose index scan的要求,執(zhí)行計(jì)劃的extra顯示using index for group-by,執(zhí)行效果等值于

SELECT MAX(log_time) FROM log_table WHERE log_machine IN (‘Machine 1')
Union
SELECT MAX(log_time) FROM log_table WHERE log_machine IN (‘Machine 2')
…..

即對每個(gè)log_machine執(zhí)行l(wèi)oose index scan,rows從原來的82636下降為16(該表總共1,000,000條記錄)。

Group by何時(shí)使用loose index scan?

適用條件:

1  針對單表操作
2  Group by使用索引的最左前綴列
3  只支持聚集函數(shù)min()/max()
4  Where條件出現(xiàn)的列必須為=constant操作 , 沒出現(xiàn)在group by中的索引列必須使用constant
5  不支持前綴索引,即部分列索引 ,如index(c1(10))
執(zhí)行計(jì)劃的extra應(yīng)該顯示using index for group-by
假定表t1有個(gè)索引idx(c1,c2,c3)

SELECT c1, c2 FROM t1 GROUP BY c1, c2;
SELECT DISTINCT c1, c2 FROM t1;
SELECT c1, MIN(c2) FROM t1 GROUP BY c1;
SELECT c1, c2 FROM t1 WHERE c1 < const GROUP BY c1, c2;
SELECT MAX(c3), MIN(c3), c1, c2 FROM t1 WHERE c2 > const GROUP BY c1, c2;
SELECT c2 FROM t1 WHERE c1 < const GROUP BY c1, c2;
SELECT c1, c2 FROM t1 WHERE c3 = const GROUP BY c1, c2
SELECT c1, c3 FROM t1 GROUP BY c1, c2;--無法使用松散索引

而SELECT c1, c3 FROM t1  where c3= const GROUP BY c1, c2;則可以

緊湊索引掃描tight index scan
Group by在無法使用loose index scan,還可以選擇tight,若兩者都不可選,則只能借助臨時(shí)表;
掃描索引時(shí),須讀取所有滿足條件的索引鍵,要么是全索引掃描,要么是范圍索引掃描;
Group by的索引列不連續(xù);或者不是從最左前綴開始,但是where條件里出現(xiàn)最左列;

SELECT c1, c2, c3 FROM t1 WHERE c2 = 'a' GROUP BY c1, c3;
SELECT c1, c2, c3 FROM t1 WHERE c1 = 'a' GROUP BY c2, c3;

5.6的改進(jìn)
事實(shí)上,5.6的index condition push down可以彌補(bǔ)loose index scan缺失帶來的性能損失。
KEY(age,zip)

mysql> explain SELECT name FROM people WHERE age BETWEEN 18 AND 20 AND zip IN (12345,12346, 12347);
+----+-------------+--------+-------+---------------+------+---------+------+-------+-------------+
| id | select_type | table | type | possible_keys | key | key_len | ref | rows | Extra    |
+----+-------------+--------+-------+---------------+------+---------+------+-------+-------------+
| 1 | SIMPLE   | people | range | age      | age | 4    | NULL | 90556 | Using where |
+----+-------------+--------+-------+---------------+------+---------+------+-------+-------------+
1 row in set (0.01 sec)

根據(jù)key_len=4可以推測出sql只用到索引的第一列,即先通過索引查出滿足age (18,20)的行記錄,然后從server層篩選出滿足zip約束的行;
pre-5.6,對于復(fù)合索引,只有當(dāng)引導(dǎo)列使用"="時(shí)才有機(jī)會(huì)在索引掃描時(shí)使用到后面的索引列。

mysql> explain SELECT name FROM people WHERE age=18 AND zip IN (12345,12346, 12347);
+----+-------------+--------+-------+---------------+------+---------+------+------+-------------+
| id | select_type | table | type | possible_keys | key | key_len | ref | rows | Extra    |
+----+-------------+--------+-------+---------------+------+---------+------+------+-------------+
| 1 | SIMPLE   | people | range | age      | age | 8    | NULL |  3 | Using where |
+----+-------------+--------+-------+---------------+------+---------+------+------+-------------+
1 row in set (0.00 sec)

對比一下查詢效率

mysql> SELECT sql_no_cache name FROM people WHERE age=19 AND zip IN (12345,12346, 12347);
+----------------------------------+
| name               |
+----------------------------------+
| 888ba838661aff00bbbce114a2a22423 |
+----------------------------------+
1 row in set (0.06 sec)
mysql> SELECT SQL_NO_CACHE name FROM people WHERE age BETWEEN 18 AND 22 AND zip IN (12345,12346, 12347);
+----------------------------------+
| name               |
+----------------------------------+
| ed4481336eb9adca222fd404fa15658e |
| 888ba838661aff00bbbce114a2a22423 |
+----------------------------------+
2 rows in set (1 min 56.09 sec)

對于第二條sql,可以使用union改寫,

mysql> SELECT name FROM people WHERE age=18 AND zip IN (12345,12346, 12347)
  -> UNION ALL
  -> SELECT name FROM people WHERE age=19 AND zip IN (12345,12346, 12347)
  -> UNION ALL
  -> SELECT name FROM people WHERE age=20 AND zip IN (12345,12346, 12347)
  -> UNION ALL
  -> SELECT name FROM people WHERE age=21 AND zip IN (12345,12346, 12347)
  -> UNION ALL
-> SELECT name FROM people WHERE age=22 AND zip IN (12345,12346, 12347);

而mysql5.6引入了index condition pushdown,從優(yōu)化器層面解決了此類問題。

相關(guān)文章

  • Mysql主從復(fù)制注意事項(xiàng)的講解

    Mysql主從復(fù)制注意事項(xiàng)的講解

    今天小編就為大家分享一篇關(guān)于Mysql主從復(fù)制注意事項(xiàng)的講解,小編覺得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來看看吧
    2019-02-02
  • Mysql字段為null的加減乘除運(yùn)算方式

    Mysql字段為null的加減乘除運(yùn)算方式

    這篇文章主要介紹了Mysql字段為null的加減乘除運(yùn)算方式,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-09-09
  • 淺談MySQL函數(shù)

    淺談MySQL函數(shù)

    這篇文章主要介紹MySQL函數(shù),主要解說數(shù)學(xué)函數(shù)、字符串函數(shù)、 時(shí)間函數(shù)、加密函數(shù),需要的朋友可以參考下面文章具體內(nèi)容
    2021-09-09
  • 在sql中實(shí)現(xiàn)取一行最大值或者最小值

    在sql中實(shí)現(xiàn)取一行最大值或者最小值

    這篇文章主要介紹了在sql中實(shí)現(xiàn)取一行最大值或者最小值,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-12-12
  • MySQL與Mongo簡單的查詢實(shí)例代碼

    MySQL與Mongo簡單的查詢實(shí)例代碼

    本文通過一個(gè)實(shí)例給大家用MySQL和mongodb分別寫一個(gè)查詢,本文圖片并茂給大家介紹的非常詳細(xì),感興趣的朋友參考下吧
    2016-10-10
  • MySQL通過存儲(chǔ)過程來添加和刪除分區(qū)的過程(List分區(qū))

    MySQL通過存儲(chǔ)過程來添加和刪除分區(qū)的過程(List分區(qū))

    這篇文章主要介紹了MySQL-通過存儲(chǔ)過程來添加和刪除分區(qū)(List分區(qū)),本文通過創(chuàng)建存儲(chǔ)過程來添加和刪除分區(qū),可以避免在分區(qū)存在時(shí)添加分區(qū)報(bào)錯(cuò),或者分區(qū)不存在時(shí)刪除分區(qū)報(bào)錯(cuò)的問題,需要的朋友可以參考下
    2023-09-09
  • 簡單介紹下MYSQL的索引類型

    簡單介紹下MYSQL的索引類型

    本文介紹了七種MySQL索引類型。在數(shù)據(jù)庫表中,對字段建立索引可以大大提高查詢速度。通過善用這些索引,可以令MySQL的查詢和運(yùn)行更加高效。
    2015-07-07
  • 最新MySQL高級(jí)SQL語句大全

    最新MySQL高級(jí)SQL語句大全

    這篇文章主要介紹了MySQL高級(jí)SQL語句,主要包括常用查詢知識(shí)order by按關(guān)鍵字排序的sql語句,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2022-06-06
  • mysql出現(xiàn)ERROR 1819 (HY000)的解決方法

    mysql出現(xiàn)ERROR 1819 (HY000)的解決方法

    這篇文章主要為大家詳細(xì)介紹了mysql出現(xiàn)ERROR 1819 (HY000)的解決方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2016-10-10
  • mysql表優(yōu)化、分析、檢查和修復(fù)的方法詳解

    mysql表優(yōu)化、分析、檢查和修復(fù)的方法詳解

    這篇文章主要介紹了mysql表優(yōu)化、分析、檢查和修復(fù)的方法,結(jié)合實(shí)例形式較為詳細(xì)的分析了MySQL表進(jìn)行優(yōu)化,分析與修復(fù)等操作的各種常見命令與使用技巧,需要的朋友可以參考下
    2016-04-04

最新評論