MySQL 8 新特性之Invisible Indexes
背景
索引是把雙刃劍,在提升查詢速度的同時會減慢DML的操作。畢竟,索引的維護需要一定的成本。所以,對于索引,要加上該加的,刪除無用的。前者是加法,后者是減法。但在實際工作中,大家似乎更熱衷于前者,而很少進行后者。究其原因,在于后者,難。難的不是操作本身,而是如何確認一個索引是無用的。
如何確認無用索引
在不可見索引出現(xiàn)之前,大家可以通過sys.schema_unused_indexes來確定無用索引。在MySQL 5.6中,即使沒有sys庫,也可通過該視圖的基表來進行查詢。
mysql> show create table sys.schema_unused_indexes\G
*************************** 1. row ***************************
View: schema_unused_indexes
Create View: CREATE ALGORITHM=MERGE DEFINER=`mysql.sys`@`localhost` SQL SECURITY INVOKER VIEW `sys`.`schema_unused_indexes` (
`object_schema`,`object_name`,`index_name`) AS select `t`.`OBJECT_SCHEMA` AS `object_schema`,`t`.`OBJECT_NAME` AS `object_name`,`t`.`INDEX_NAME` AS `index_name` from (`performance_schema`.`table_io_waits_summary_by_index_usage` `t` join `information_schema`.`STATISTICS` `s` on(((`t`.`OBJECT_SCHEMA` = convert(`s`.`TABLE_SCHEMA` using utf8mb4)) and (`t`.`OBJECT_NAME` = convert(`s`.`TABLE_NAME` using utf8mb4)) and (convert(`t`.`INDEX_NAME` using utf8) = `s`.`INDEX_NAME`)))) where ((`t`.`INDEX_NAME` is not null) and (`t`.`COUNT_STAR` = 0) and (`t`.`OBJECT_SCHEMA` <> 'mysql') and (`t`.`INDEX_NAME` <> 'PRIMARY') and (`s`.`NON_UNIQUE` = 1) and (`s`.`SEQ_IN_INDEX` = 1)) order by `t`.`OBJECT_SCHEMA`,`t`.`OBJECT_NAME`character_set_client: utf8mb4
collation_connection: utf8mb4_0900_ai_ci
1 row in set, 1 warning (0.00 sec)
但這種方式也有不足,
1. 如果實例發(fā)生重啟,performance_schema中的數(shù)據(jù)就會清零。
2. 如果基于上面的查詢刪除了索引,查詢性能突然變差,怎么辦?
不可見索引的出現(xiàn),可有效彌補上述不足。將index設(shè)置為invisible,會導致優(yōu)化器在選擇執(zhí)行計劃時,自動忽略該索引,即便使用了FORCE INDEX。
當然,這個是由optimizer_switch變量中use_invisible_indexes選項決定的,默認為off。如果想看一個查詢在索引調(diào)整前后執(zhí)行計劃的差別,可在會話級別調(diào)整use_invisible_indexes的值,如,
mysql> show create table slowtech.t1\G *************************** 1. row *************************** Table: t1 Create Table: CREATE TABLE `t1` ( `id` int(11) NOT NULL, `name` varchar(10) DEFAULT NULL, PRIMARY KEY (`id`), KEY `idx_name` (`name`) /*!80000 INVISIBLE */ ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci 1 row in set (0.00 sec) mysql> explain select * from slowtech.t1 where name='a'; +----+-------------+-------+------------+------+---------------+------+---------+------+------+----------+-------------+ | id | select_type | table | partitions | type | possible_keys | key | key_len | ref | rows | filtered | Extra | +----+-------------+-------+------------+------+---------------+------+---------+------+------+----------+-------------+ | 1 | SIMPLE | t1 | NULL | ALL | NULL | NULL | NULL | NULL | 6 | 16.67 | Using where | +----+-------------+-------+------------+------+---------------+------+---------+------+------+----------+-------------+ 1 row in set, 1 warning (0.00 sec) mysql> set session optimizer_switch="use_invisible_indexes=on"; Query OK, 0 rows affected (0.00 sec) mysql> explain select * from slowtech.t1 where name='a'; +----+-------------+-------+------------+------+---------------+----------+---------+-------+------+----------+-------------+ | id | select_type | table | partitions | type | possible_keys | key | key_len | ref | rows | filtered | Extra | +----+-------------+-------+------------+------+---------------+----------+---------+-------+------+----------+-------------+ | 1 | SIMPLE | t1 | NULL | ref | idx_name | idx_name | 43 | const | 1 | 100.00 | Using index | +----+-------------+-------+------------+------+---------------+----------+---------+-------+------+----------+-------------+ 1 row in set, 1 warning (0.00 sec)
不可見索引的常見操作
create table t1(id int primary key,name varchar(10),index idx_name (name) invisible); alter table t1 alter index idx_name visible; alter table t1 alter index idx_name invisible;
如何查看哪些索引不可見
mysql> select table_schema,table_name,index_name,column_name,is_visible from information_schema.statistics where is_visible='no'; +--------------+------------+------------+-------------+------------+ | TABLE_SCHEMA | TABLE_NAME | INDEX_NAME | COLUMN_NAME | IS_VISIBLE | +--------------+------------+------------+-------------+------------+ | slowtech | t1 | idx_name | name | NO | +--------------+------------+------------+-------------+------------+ 1 row in set (0.00 sec)
注意
1. 主鍵索引不可被設(shè)置為invisible。
總結(jié)
以上所述是小編給大家介紹的MySQL 8 新特性之Invisible Indexes ,希望對大家有所幫助,如果大家有任何疑問歡迎給我留言,小編會及時回復大家的!
- MySQL如何基于Explain關(guān)鍵字優(yōu)化索引功能
- MySQL中or、in、union與索引優(yōu)化詳析
- MySQL中索引優(yōu)化distinct語句及distinct的多字段操作
- 探究MySQL優(yōu)化器對索引和JOIN順序的選擇
- MySQL的id關(guān)聯(lián)和索引使用的實際優(yōu)化案例
- Mysql limit 優(yōu)化,百萬至千萬級快速分頁 復合索引的引用并應用于輕量級框架
- Mysql limit 優(yōu)化,百萬至千萬級快速分頁 復合索引的引用并應用于輕量級框架
- MySQL8.0?索引優(yōu)化invisible?index詳情
相關(guān)文章
windows下如何解決mysql secure_file_priv null問題
這篇文章主要介紹了windows下如何解決mysql secure_file_priv null問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2024-01-01
mysql如何將數(shù)據(jù)庫中的所有表結(jié)構(gòu)和數(shù)據(jù)導入到另一個庫
介紹了如何使用mysqldump命令備份和導入數(shù)據(jù)庫,以及創(chuàng)建目標數(shù)據(jù)庫的步驟,首先使用mysqldump備份源數(shù)據(jù)庫,然后在目標數(shù)據(jù)庫中創(chuàng)建數(shù)據(jù)庫,并將備份文件導入到目標數(shù)據(jù)庫,確保數(shù)據(jù)結(jié)構(gòu)和內(nèi)容完整復制,提到了DataGrip、Navicat在導入導出過程中可能出現(xiàn)的問題2024-10-10
淺談Mysql中類似于nvl()函數(shù)的ifnull()函數(shù)
下面小編就為大家?guī)硪黄獪\談Mysql中類似于nvl()函數(shù)的ifnull()函數(shù)。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-02-02

