(譯)MySQL 8.0實驗室---MySQL中的倒序索引(Descending Indexes)

 

譯者注:
MySQL 8.0以前,不論是否指定索引建的排序方式,都會忽略建立索引時候指定的排序方式(語法上不會報錯),最終都會建立爲ASC方式的索引,
在執行查詢的時候,只存在forwarded(正向)方式對索引進行掃描。
關於正向索引和反向索引,邏輯上很容易理解,這裏有兩個相關的概念:
正向索引或者反向(倒序)索引,二者都是在構建B樹索引時候的相關字段排序方式,是B索引樹的邏輯存儲方式
正向掃描(forward)和反向掃描( Backward index scan;)是執行查詢的過程當中對B樹索引的掃描方式,是數據執行計劃時候的一種索引掃描方式
關於正向掃描或者反向掃描不是隨意的,受sql語句中(正/反向)排序方式以及(正/反向)索引的影響
以前在sqlserver中簡單寫過一點相似的東西,http://www.javashuo.com/article/p-tnibdrio-gz.htmlhtml

總體上看,拋開正向索引和倒序索引,在掃描掃描的過程當中,正向索引掃描的在性能上,稍微優於反向索引掃描。
不過,即使是反向索引掃描,也是優化器根據具體查詢進行優化的結果,並不是一個很差的選擇。mysql

 

 


原文連接:http://mysqlserverteam.com/mysql-8-0-labs-descending-indexes-in-mysql/ sql

 

如下爲譯文:sqlserver

從8.0優化器實驗室發佈開始,MySQL開始支持倒序索引。
正如我將在本文中詳細介紹的,這個新特性能夠用來消除對結果排序的需求,並在許多查詢中帶來性能改進。性能

 

簡介測試

在此版本以前,全部索引都是按升序建立的。當語法自己被解析時,元數據不會被保留。例如在MySQL 5.7中:優化

mysql 5.7> CREATE TABLE t1 (a INT, b INT, INDEX a_desc_b_asc (a DESC, b ASC));
Query OK, 0 rows affected (0.47 sec)

mysql 5.7> SHOW CREATE TABLE t1\G
*************************** 1. row ***************************
       Table: t1
Create Table: CREATE TABLE `t1` (
  `a` int(11) DEFAULT NULL,
  `b` int(11) DEFAULT NULL,
  KEY `a_desc_b_asc` (`a`,`b`) <-- 建立索引時候的元數據沒有被保留
) ENGINE=InnoDB DEFAULT CHARSET=latin1
1 row in set (0.00 sec)

應該注意的是,MySQL 5.7 optimizer可以反向掃描一個升序索引(按照降序排列),其成本較高spa

(譯者注:以上是原文中寫道的,MySQL 5.7中不知道怎麼去判斷在對索引掃描的時候,到底是正向掃描仍是反向掃描)。
以下能夠進一步測試,咱們能夠看到正向索引掃描比反向索引掃描好~15%。
不能支持倒敘索引的主要限制是,優化器必須對混合順序(如DESC、b ASC的順序)使用文件排序。code

MySQL 8.0中的改進server

引入反向索引後,InnoDB如今能夠按照降序順序存儲數據行,優化器將在查詢中請求降序時利用它。
重複上面的例子,咱們能夠看到在建立表時索引順序信息被正確地保留了:

mysql 8.0> CREATE TABLE t1 (a INT, b INT, INDEX a_desc_b_asc (a DESC, b ASC));
Query OK, 0 rows affected (0.47 sec)
 
mysql 8.0> show create table t1;
+-------+--------------------------------------------------------------------------------------------------------------------------------------------------------+
| Table | Create Table |
+-------+--------------------------------------------------------------------------------------------------------------------------------------------------------+
| t1 | CREATE TABLE `t1` (
`a` int(11) DEFAULT NULL,
`b` int(11) DEFAULT NULL,
KEY `a_desc_b_asc` (`a` DESC,`b`)
) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
+-------+--------------------------------------------------------------------------------------------------------------------------------------------------------+
1 row in set (0.00 sec)

爲了區分向後和向前索引掃描,還改進了EXPLAIN的輸出。
對於MySQL-5.7,除了查詢2和查詢6以外,咱們對全部查詢都使用反向索引掃描或文件排序,由於這兩個查詢只須要升序。

 

Query 1: SELECT * FROM t1 ORDER BY a DESC;

mysql 8.0> explain SELECT * FROM t1 ORDER BY a DESC;
+----+-------------+-------+------------+-------+---------------+--------------+---------+------+------+----------+-------------+
| id | select_type | table | partitions | type | possible_keys | key | key_len | ref | rows | filtered | Extra |
+----+-------------+-------+------------+-------+---------------+--------------+---------+------+------+----------+-------------+
| 1  | SIMPLE    | t1    | NULL    | index  | NULL | a_desc_b_asc | 10 | NULL | 10 | 100.00 | Using index |
+----+-------------+-------+------------+-------+---------------+--------------+---------+------+------+----------+-------------+
1 row in set, 1 warning (0.00 sec)

Query 2: SELECT * FROM t1 ORDER BY a ASC;

mysql 8.0> explain SELECT * FROM t1 ORDER BY a ASC;
+----+-------------+-------+------------+-------+---------------+--------------+---------+------+------+----------+----------------------------------+
| id | select_type | table | partitions | type | possible_keys | key | key_len | ref | rows | filtered | Extra |
+----+-------------+-------+------------+-------+---------------+--------------+---------+------+------+----------+----------------------------------+
| 1 | SIMPLE | t1 | NULL | index | NULL | a_desc_b_asc | 10 | NULL | 10 | 100.00 | Backward index scan; Using index |
+----+-------------+-------+------------+-------+---------------+--------------+---------+------+------+----------+----------------------------------+
1 row in set, 1 warning (0.00 sec)

Query 3: SELECT * FROM t1 ORDER BY a DESC, b ASC;

mysql 8.0> EXPLAIN SELECT * FROM t1 ORDER BY a DESC, b ASC;
+----+-------------+-------+------------+-------+---------------+--------------+---------+------+------+----------+-------------+
| id | select_type | table | partitions | type | possible_keys | key | key_len | ref | rows | filtered | Extra |
+----+-------------+-------+------------+-------+---------------+--------------+---------+------+------+----------+-------------+
| 1 | SIMPLE | t1 | NULL | index | NULL | a_desc_b_asc | 10 | NULL | 10 | 100.00 | Using index |
+----+-------------+-------+------------+-------+---------------+--------------+---------+------+------+----------+-------------+
1 row in set, 1 warning (0.00 sec)

Query 4: SELECT * FROM t1 ORDER BY a ASC, b DESC;

相關文章
相關標籤/搜索