首頁  >  文章  >  資料庫  >  MySQL查詢效能最佳化的索引潛水實例分析

MySQL查詢效能最佳化的索引潛水實例分析

WBOY
WBOY轉載
2023-06-03 12:09:25657瀏覽

先要從一件怪事說起:

我先造點資料複現一下問題,建立一張使用者表:

CREATE TABLE `user` (
  `id` bigint(20) unsigned NOT NULL AUTO_INCREMENT COMMENT '主键ID',
  `name` varchar(100) NOT NULL DEFAULT '' COMMENT '姓名',
  `age` int(11) NOT NULL DEFAULT 0 COMMENT '年龄',
  PRIMARY KEY (`id`),
  KEY `idx_age` (`age`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;

透過一批使用者年齡,查詢該年齡的使用者信息,並查看SQL執行計劃:

explain select * from user 
where age in (1,2,3,4,5,6,7,8,9);

MySQL查詢效能最佳化的索引潛水實例分析

where條件中有9個參數,重點在於執行計畫中的預估掃描行數為279行。

到這裡沒什麼問題,預估的非常準,實際就是279行。

MySQL查詢效能最佳化的索引潛水實例分析

但是,問題來了,當我們在where條件中,再加一個參數,變成了10個參數,預估掃描行數本來應該增加,結果卻大大減少了。

explain select * from user
where age in (1,2,3,4,5,6,7,8,9,10);

MySQL查詢效能最佳化的索引潛水實例分析

一下子減少到了30行,可是實際行數是多少呢?

MySQL查詢效能最佳化的索引潛水實例分析

實際上是310行,預估掃描行數是30行,真是錯到姥姥家了。

MySQL咋回事啊,到底還能不能預估?

不能預估的話,換其他人!

大家一定也是滿臉疑惑,直到我去官網上看到了一個詞語,索引潛水(Index dive) 。

跟這個字相關的,還有一個設定參數 eq_range_index_dive_limit

MySQL5.7.3之前的版本,這個值預設是10,之後的版本,這個值預設是200。

可以使用指令來看看這個值的大小:

show variables like '%eq_range_index_dive_limit%';

MySQL查詢效能最佳化的索引潛水實例分析

#當然,我們也可以手動修改這個值的大小:

set eq_range_index_dive_limit=200;

這個 eq_range_index_dive_limit 配置的功能是:

當where語句in條件中參數數小於這個值的時候,MySQL就採用索引潛水(Index dive) 的方式預估掃描行數,非常精確。

當where語句in條件中參數個數大於等於這個值的時候,MySQL就採用另一種方​​式索引統計(Index statistics) 預估掃描行數,誤差較大。

MySQL為什麼要這麼做呢?

都用索引潛水(Index dive) 的方式預估掃描行數,不好嗎?

其實這是基於成本的考慮,索引潛水估算成本較高,適合小數據量。 索引統計估算成本較低,適合大資料量。

一般情況下,我們的where語句的in條件的參數不會太多,適合使用索引潛水預估掃描行數。

建議還在使用MySQL5.7.3之前版本的同學們,手動修改一下索引潛水的配置參數,改成合適的數值。

如果你們專案中in條件最多有500個參數,就把配置參數改成501。

這樣MySQL預估掃描行數比較準確,可以選擇更適合的索引。

以上是MySQL查詢效能最佳化的索引潛水實例分析的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文轉載於:yisu.com。如有侵權,請聯絡admin@php.cn刪除