mysql索引合併:一條sql可以使用多個索引
前言
mysql的索引合併並不是什麼新特性。早在mysql5.0版本就已經實現。之所以還寫這篇博文,是因為好多人還一直保留著一條sql語句只能使用一個索引的錯誤觀念。本文會透過一些示例來說明如何使用索引合併。
什麼是索引合併
下面我們看下mysql文件中對索引合併的說明:
The Index Merge method is used to retrieve rows with several range scans and to merge their results into one. The merge can produce unions, intersections, or unions-of-intersections of its underlying scans. This access method merges index scans from a single table; it does not merge scans across multiple tables.
根據官方文件中的說明,我們可以瞭解到:
1、索引合併是把幾個索引的範圍掃描合併成一個索引。
2、索引合併的時候,會對索引進行並集,交集或者先交集再並集操作,以便合併成一個索引。
3、這些需要合併的索引只能是一個表的。不能對多表進行索引合併。
怎麼確定使用了索引合併
在使用explain對sql語句進行操作時,如果使用了索引合併,那麼在輸出內容的type列會顯示 index_merge,key列會顯示出所有使用的索引。如下:
index_merge_sql
使用索引合併的示例
資料表結構
mysql> show create table test\G
*************************** 1. row ***************************
Table: test
Create Table: CREATE TABLE `test` (
`id` int(11) NOT NULL AUTO_INCREMENT,
`key1_part1` int(11) NOT NULL DEFAULT '0',
`key1_part2` int(11) NOT NULL DEFAULT '0',
`key2_part1` int(11) NOT NULL DEFAULT '0',
`key2_part2` int(11) NOT NULL DEFAULT '0',
PRIMARY KEY (`id`),
KEY `key1` (`key1_part1`,`key1_part2`),
KEY `key2` (`key2_part1`,`key2_part2`)
) ENGINE=MyISAM AUTO_INCREMENT=18 DEFAULT CHARSET=utf8
1 row in set (0.00 sec)
資料
mysql> select * from test;
+----+------------+------------+------------+------------+
| id | key1_part1 | key1_part2 | key2_part1 | key2_part2 |
+----+------------+------------+------------+------------+
| 1 | 1 | 1 | 1 | 1 |
| 2 | 1 | 1 | 2 | 1 |
| 3 | 1 | 1 | 2 | 2 |
| 4 | 1 | 1 | 3 | 2 |
| 5 | 1 | 1 | 3 | 3 |
| 6 | 1 | 1 | 4 | 3 |
| 7 | 1 | 1 | 4 | 4 |
| 8 | 1 | 1 | 5 | 4 |
| 9 | 1 | 1 | 5 | 5 |
| 10 | 2 | 1 | 1 | 1 |
| 11 | 2 | 2 | 1 | 1 |
| 12 | 3 | 2 | 1 | 1 |
| 13 | 3 | 3 | 1 | 1 |
| 14 | 4 | 3 | 1 | 1 |
| 15 | 4 | 4 | 1 | 1 |
| 16 | 5 | 4 | 1 | 1 |
| 17 | 5 | 5 | 1 | 1 |
| 18 | 5 | 5 | 3 | 3 |
| 19 | 5 | 5 | 3 | 1 |
| 20 | 5 | 5 | 3 | 2 |
| 21 | 5 | 5 | 3 | 4 |
| 22 | 6 | 6 | 3 | 3 |
| 23 | 6 | 6 | 3 | 4 |
| 24 | 6 | 6 | 3 | 5 |
| 25 | 6 | 6 | 3 | 6 |
| 26 | 6 | 6 | 3 | 7 |
| 27 | 1 | 1 | 3 | 6 |
| 28 | 1 | 2 | 3 | 6 |
| 29 | 1 | 3 | 3 | 6 |
+----+------------+------------+------------+------------+
29 rows in set (0.00 sec)
使用索引合併的案例
mysql> explain select * from test where (key1_part1=4 and key1_part2=4) or key2_part1=4\G
*************************** 1. row ***************************
id: 1
select_type: SIMPLE
table: test
type: index_merge
possible_keys: key1,key2
key: key1,key2
key_len: 8,4
ref: NULL
rows: 3
Extra: Using sort_union(key1,key2); Using where
1 row in set (0.00 sec)
未使用索引合併的案例
mysql> explain select * from test where (key1_part1=1 and key1_part2=1) or key2_part1=4\G
*************************** 1. row ***************************
id: 1
select_type: SIMPLE
table: test
type: ALL
possible_keys: key1,key2
key: NULL
key_len: NULL
ref: NULL
rows: 29
Extra: Using where
1 row in set (0.00 sec)
從上面的兩個案例大家可以發現,相同模式的sql語句,可能有時能使用索引,有時不能使用索引。是否能使用索引,取決於mysql查詢最佳化器對統計資料分析後,是否認為使用索引更快。
因此,單純的討論一條sql是否可以使用索引有點片面,還需要考慮資料。
注意事項
mysql5.6.7之前的版本遵守range優先的原則。也就是說,當一個索引的一個連續段,包含所有符合查詢要求的資料時,哪怕索引合併能提供效率,也不再使用索引合併。舉個例子:
mysql> explain select * from test where (key1_part1=1 and key1_part2=1) and key2_part1=1\G
*************************** 1. row ***************************
id: 1
select_type: SIMPLE
table: test
type: ref
possible_keys: key1,key2
key: key2
key_len: 4
ref: const
rows: 9
Extra: Using where
1 row in set (0.00 sec)
上面符合查詢要求的結果只有一條,而這一條記錄被索引key2所包含。
可以看到這條sql語句使用了key2索引。但是這個並不是最快的執行方式。其實,把索引key1和索引key2進行索引合併,取交集後,就發現只有一條記錄適合。應該查詢效率會更快。
tips:這條sql語句未在mysql5.6.7之後版本執行驗證,以上為理論推導。有興趣的話,您可以到mysql5.6.7之後版本上驗證下。
轉自:%e7%b4%a2%e5%bc%95%e5%90%88%e5%b9%b6%e4%b8%80%e6%9d%a1sql%e5%8f%af%e4%bb%a5%e4%bd%bf%e7%94%a8%e5%a4%9a%e4%b8%aa%e7%b4%a2%e5%bc%95/
mysql的索引合併並不是什麼新特性。早在mysql5.0版本就已經實現。之所以還寫這篇博文,是因為好多人還一直保留著一條sql語句只能使用一個索引的錯誤觀念。本文會透過一些示例來說明如何使用索引合併。
什麼是索引合併
下面我們看下mysql文件中對索引合併的說明:
The Index Merge method is used to retrieve rows with several range scans and to merge their results into one. The merge can produce unions, intersections, or unions-of-intersections of its underlying scans. This access method merges index scans from a single table; it does not merge scans across multiple tables.
根據官方文件中的說明,我們可以瞭解到:
1、索引合併是把幾個索引的範圍掃描合併成一個索引。
2、索引合併的時候,會對索引進行並集,交集或者先交集再並集操作,以便合併成一個索引。
3、這些需要合併的索引只能是一個表的。不能對多表進行索引合併。
怎麼確定使用了索引合併
在使用explain對sql語句進行操作時,如果使用了索引合併,那麼在輸出內容的type列會顯示 index_merge,key列會顯示出所有使用的索引。如下:
index_merge_sql
使用索引合併的示例
資料表結構
mysql> show create table test\G
*************************** 1. row ***************************
Table: test
Create Table: CREATE TABLE `test` (
`id` int(11) NOT NULL AUTO_INCREMENT,
`key1_part1` int(11) NOT NULL DEFAULT '0',
`key1_part2` int(11) NOT NULL DEFAULT '0',
`key2_part1` int(11) NOT NULL DEFAULT '0',
`key2_part2` int(11) NOT NULL DEFAULT '0',
PRIMARY KEY (`id`),
KEY `key1` (`key1_part1`,`key1_part2`),
KEY `key2` (`key2_part1`,`key2_part2`)
) ENGINE=MyISAM AUTO_INCREMENT=18 DEFAULT CHARSET=utf8
1 row in set (0.00 sec)
資料
mysql> select * from test;
+----+------------+------------+------------+------------+
| id | key1_part1 | key1_part2 | key2_part1 | key2_part2 |
+----+------------+------------+------------+------------+
| 1 | 1 | 1 | 1 | 1 |
| 2 | 1 | 1 | 2 | 1 |
| 3 | 1 | 1 | 2 | 2 |
| 4 | 1 | 1 | 3 | 2 |
| 5 | 1 | 1 | 3 | 3 |
| 6 | 1 | 1 | 4 | 3 |
| 7 | 1 | 1 | 4 | 4 |
| 8 | 1 | 1 | 5 | 4 |
| 9 | 1 | 1 | 5 | 5 |
| 10 | 2 | 1 | 1 | 1 |
| 11 | 2 | 2 | 1 | 1 |
| 12 | 3 | 2 | 1 | 1 |
| 13 | 3 | 3 | 1 | 1 |
| 14 | 4 | 3 | 1 | 1 |
| 15 | 4 | 4 | 1 | 1 |
| 16 | 5 | 4 | 1 | 1 |
| 17 | 5 | 5 | 1 | 1 |
| 18 | 5 | 5 | 3 | 3 |
| 19 | 5 | 5 | 3 | 1 |
| 20 | 5 | 5 | 3 | 2 |
| 21 | 5 | 5 | 3 | 4 |
| 22 | 6 | 6 | 3 | 3 |
| 23 | 6 | 6 | 3 | 4 |
| 24 | 6 | 6 | 3 | 5 |
| 25 | 6 | 6 | 3 | 6 |
| 26 | 6 | 6 | 3 | 7 |
| 27 | 1 | 1 | 3 | 6 |
| 28 | 1 | 2 | 3 | 6 |
| 29 | 1 | 3 | 3 | 6 |
+----+------------+------------+------------+------------+
29 rows in set (0.00 sec)
使用索引合併的案例
mysql> explain select * from test where (key1_part1=4 and key1_part2=4) or key2_part1=4\G
*************************** 1. row ***************************
id: 1
select_type: SIMPLE
table: test
type: index_merge
possible_keys: key1,key2
key: key1,key2
key_len: 8,4
ref: NULL
rows: 3
Extra: Using sort_union(key1,key2); Using where
1 row in set (0.00 sec)
未使用索引合併的案例
mysql> explain select * from test where (key1_part1=1 and key1_part2=1) or key2_part1=4\G
*************************** 1. row ***************************
id: 1
select_type: SIMPLE
table: test
type: ALL
possible_keys: key1,key2
key: NULL
key_len: NULL
ref: NULL
rows: 29
Extra: Using where
1 row in set (0.00 sec)
從上面的兩個案例大家可以發現,相同模式的sql語句,可能有時能使用索引,有時不能使用索引。是否能使用索引,取決於mysql查詢最佳化器對統計資料分析後,是否認為使用索引更快。
因此,單純的討論一條sql是否可以使用索引有點片面,還需要考慮資料。
注意事項
mysql5.6.7之前的版本遵守range優先的原則。也就是說,當一個索引的一個連續段,包含所有符合查詢要求的資料時,哪怕索引合併能提供效率,也不再使用索引合併。舉個例子:
mysql> explain select * from test where (key1_part1=1 and key1_part2=1) and key2_part1=1\G
*************************** 1. row ***************************
id: 1
select_type: SIMPLE
table: test
type: ref
possible_keys: key1,key2
key: key2
key_len: 4
ref: const
rows: 9
Extra: Using where
1 row in set (0.00 sec)
上面符合查詢要求的結果只有一條,而這一條記錄被索引key2所包含。
可以看到這條sql語句使用了key2索引。但是這個並不是最快的執行方式。其實,把索引key1和索引key2進行索引合併,取交集後,就發現只有一條記錄適合。應該查詢效率會更快。
tips:這條sql語句未在mysql5.6.7之後版本執行驗證,以上為理論推導。有興趣的話,您可以到mysql5.6.7之後版本上驗證下。
轉自:%e7%b4%a2%e5%bc%95%e5%90%88%e5%b9%b6%e4%b8%80%e6%9d%a1sql%e5%8f%af%e4%bb%a5%e4%bd%bf%e7%94%a8%e5%a4%9a%e4%b8%aa%e7%b4%a2%e5%bc%95/
來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/28939273/viewspace-1991212/,如需轉載,請註明出處,否則將追究法律責任。
相關文章
- mysql 索引合併MySql索引
- Mysql索引的使用-組合索引+跳躍條件MySql索引
- 唯一索引,可以在索引列插入多個null嗎索引Null
- Mysql索引的使用 - 組合索引 + 範圍條件的處理MySql索引
- 聯合索引和多個單列索引使用中的索引命中情況及索引建立原則索引
- MySQL 配置索引頁的合併閾值MySql索引
- 又一個複合索引的SQL調優索引SQL
- mysql組合索引,abc索引命中MySql索引
- MySQL中為什麼要使用索引合併(Index Merge)?MySql索引Index
- MySQL如何建立一個好索引?建立索引的5條建議【宇哥帶你玩轉MySQL 索引篇(三)】MySql索引
- MySQL複合索引MySql索引
- MySql如何使用索引(一)MySql索引
- index merge合併索引Index索引
- Oracle索引合併coalesce操作Oracle索引
- MySQL 優化之 index_merge (索引合併)MySql優化Index索引
- MySQL複合索引探究MySql索引
- MySQL建立複合索引MySql索引
- MySQL的聯合索引MySql索引
- mysql的組合索引MySql索引
- SQL優化(二)(聯合索引的使用)SQL優化索引
- oracle複合索引介紹(多欄位索引)Oracle索引
- 【Mysql】InnoDB 中的聚簇索引、二級索引、聯合索引MySql索引
- 12.索引合併-indexMerge索引Index
- Mysql索引使用MySql索引
- MySQL 唯一索引和普通索引MySql索引
- SQL優化案例一則--複合索引沒有被使用SQL優化索引
- MySQL 組合索引不生效MySql索引
- MySQL學習筆記:組合索引-索引下推MySql筆記索引
- MySQL單列索引和組合索引的區別MySql索引
- MySql索引使用策略MySql索引
- Mysql——index(索引)使用MySqlIndex索引
- MySQL索引和SQL調優MySql索引
- SQL最佳化案例一則--複合索引沒有被使用SQL索引
- SQL Server 索引結構及其使用(一)SQLServer索引
- 一個複合索引的優化案例索引優化
- Mysql索引以及使用索引注意事項MySql索引
- 【Mysql】一個簡易的索引方案MySql索引
- MySQL <唯一索引>MySql索引