阿里云-云小站(无限量代金券发放中)
【腾讯云】云服务器、云数据库、COS、CDN、短信等热卖云产品特惠抢购

MySQL的索引与优化

221次阅读
没有评论

共计 9993 个字符,预计需要花费 25 分钟才能阅读完成。

写在前面:索引对查询的速度有着至关重要的影响,理解索引也是进行数据库性能调优的起点。考虑如下情况,假设数据库中一个表有 10^6 条记录,DBMS 的页面大小为 4K,并存储 100 条记录。如果没有索引,查询将对整个表进行扫描,最坏的情况下,如果所有数据页都不在内存,需要读取 10^4 个页面,如果这 10^4 个页面在磁盘上随机分布,需要进行 10^4 次 I /O,假设磁盘每次 I / O 时间为 10ms(忽略数据传输时间),则总共需要 100s(但实际上要好很多很多)。如果对之建立 B -Tree 索引,则只需要进行 log100(10^6)= 3 次页面读取,最坏情况下耗时 30ms。这就是索引带来的效果,很多时候,当你的应用程序进行 SQL 查询速度很慢时,应该想想是否可以建索引。进入正题:

第二章、索引与优化

1、选择索引的数据类型

MySQL 支持很多数据类型,选择合适的数据类型存储数据对性能有很大的影响。通常来说,可以遵循以下一些指导原则:

(1)越小的数据类型通常更好:越小的数据类型通常在磁盘、内存和 CPU 缓存中都需要更少的空间,处理起来更快。
(2)简单的数据类型更好:整型数据比起字符,处理开销更小,因为字符串的比较更复杂。在 MySQL 中,应该用内置的日期和时间数据类型,而不是用字符串来存储时间;以及用整型数据类型存储 IP 地址。
(3)尽量避免 NULL:应该指定列为 NOT NULL,除非你想存储 NULL。在 MySQL 中,含有空值的列很难进行查询优化,因为它们使得索引、索引的统计信息以及比较运算更加复杂。你应该用 0、一个特殊的值或者一个空串代替空值。

1.1、选择标识符
选择合适的标识符是非常重要的。选择时不仅应该考虑存储类型,而且应该考虑 MySQL 是怎样进行运算和比较的。一旦选定数据类型,应该保证所有相关的表都使用相同的数据类型。
(1)   整型:通常是作为标识符的最好选择,因为可以更快的处理,而且可以设置为 AUTO_INCREMENT。

(2)   字符串:尽量避免使用字符串作为标识符,它们消耗更好的空间,处理起来也较慢。而且,通常来说,字符串都是随机的,所以它们在索引中的位置也是随机的,这会导致页面分裂、随机访问磁盘,聚簇索引分裂(对于使用聚簇索引的存储引擎)。

2、索引入门
对于任何 DBMS,索引都是进行优化的最主要的因素。对于少量的数据,没有合适的索引影响不是很大,但是,当随着数据量的增加,性能会急剧下降。
如果对多列进行索引 (组合索引),列的顺序非常重要,MySQL 仅能对索引最左边的前缀进行有效的查找。例如:
假设存在组合索引 it1c1c2(c1,c2),查询语句 select * from t1 where c1=1 and c2= 2 能够使用该索引。查询语句 select * from t1 where c1= 1 也能够使用该索引。但是,查询语句 select * from t1 where c2= 2 不能够使用该索引,因为没有组合索引的引导列,即,要想使用 c2 列进行查找,必需出现 c1 等于某值。

2.1、索引的类型
索引是在存储引擎中实现的,而不是在服务器层中实现的。所以,每种存储引擎的索引都不一定完全相同,并不是所有的存储引擎都支持所有的索引类型。
2.1.1、B-Tree 索引
假设有如下一个表:

CREATE TABLE People (

  last_name varchar(50)    not null,

  first_name varchar(50)    not null,

  dob        date          not null,

  gender    enum(‘m’, ‘f’) not null,

  key(last_name, first_name, dob)

);

 其索引包含表中每一行的 last_name、first_name 和 dob 列。其结构大致如下:

MySQL 的索引与优化 

 索引存储的值按索引列中的顺序排列。可以利用 B -Tree 索引进行全关键字、关键字范围和关键字前缀查询,当然,如果想使用索引,你必须保证按索引的最左边前缀 (leftmost prefix of the index) 来进行查询。
(1)匹配全值 (Match the full value):对索引中的所有列都指定具体的值。例如,上图中索引可以帮助你查找出生于 1960-01-01 的 Cuba Allen。
(2) 匹配最左前缀 (Match a leftmost prefix):你可以利用索引查找 last name 为 Allen 的人,仅仅使用索引中的第 1 列。
(3) 匹配列前缀 (Match a column prefix):例如,你可以利用索引查找 last name 以 J 开始的人,这仅仅使用索引中的第 1 列。
(4) 匹配值的范围查询 (Match a range of values):可以利用索引查找 last name 在 Allen 和 Barrymore 之间的人,仅仅使用索引中第 1 列。
(5) 匹配部分精确而其它部分进行范围匹配 (Match one part exactly and match a range on another part):可以利用索引查找 last name 为 Allen,而 first name 以字母 K 开始的人。
(6) 仅对索引进行查询 (Index-only queries):如果查询的列都位于索引中,则不需要读取元组的值。
由于 B - 树中的节点都是顺序存储的,所以可以利用索引进行查找(找某些值),也可以对查询结果进行 ORDER BY。当然,使用 B -tree 索引有以下一些限制:
(1) 查询必须从索引的最左边的列开始。关于这点已经提了很多遍了。例如你不能利用索引查找在某一天出生的人。
(2) 不能跳过某一索引列。例如,你不能利用索引查找 last name 为 Smith 且出生于某一天的人。
(3) 存储引擎不能使用索引中范围条件右边的列。例如,如果你的查询语句为 WHERE last_name=”Smith” AND first_name LIKE ‘J%’ AND dob=’1976-12-23’,则该查询只会使用索引中的前两列,因为 LIKE 是范围查询。

2.1.2、Hash 索引
MySQL 中,只有 Memory 存储引擎显示支持 hash 索引,是 Memory 表的默认索引类型,尽管 Memory 表也可以使用 B -Tree 索引。Memory 存储引擎支持非唯一 hash 索引,这在数据库领域是罕见的,如果多个值有相同的 hash code,索引把它们的行指针用链表保存到同一个 hash 表项中。
假设创建如下一个表:
CREATE TABLE testhash (
  fname VARCHAR(50) NOT NULL,
  lname VARCHAR(50) NOT NULL,
  KEY USING HASH(fname)
) ENGINE=MEMORY;
包含的数据如下:
MySQL 的索引与优化

假设索引使用 hash函数 f(),如下:

f(‘Arjen’) = 2323

f(‘Baron’) = 7437

f(‘Peter’) = 8784

f(‘Vadim’) = 2458

此时,索引的结构大概如下:

MySQL 的索引与优化 

 Slots 是有序的,但是记录不是有序的。当你执行
mysql> SELECT lname FROM testhash WHERE fname=’Peter’;
MySQL 会计算’Peter’的 hash 值,然后通过它来查询索引的行指针。因为 f(‘Peter’) = 8784,MySQL 会在索引中查找 8784,得到指向记录 3 的指针。
因为索引自己仅仅存储很短的值,所以,索引非常紧凑。Hash 值不取决于列的数据类型,一个 TINYINT 列的索引与一个长字符串列的索引一样大。
 
Hash 索引有以下一些限制:
(1)由于索引仅包含 hash code 和记录指针,所以,MySQL 不能通过使用索引避免读取记录。但是访问内存中的记录是非常迅速的,不会对性造成太大的影响。
(2)不能使用 hash 索引排序。
(3)Hash 索引不支持键的部分匹配,因为是通过整个索引值来计算 hash 值的。
(4)Hash 索引只支持等值比较,例如使用 =,IN()和 <=>。对于 WHERE price>100 并不能加速查询。
2.1.3、空间 (R-Tree) 索引
MyISAM 支持空间索引,主要用于地理空间数据类型,例如 GEOMETRY。
2.1.4、全文(Full-text) 索引
全文索引是 MyISAM 的一个特殊索引类型,主要用于全文检索。

3、高性能的索引策略
3.1、聚簇索引(Clustered Indexes)
聚簇索引保证关键字的值相近的元组存储的物理位置也相同(所以字符串类型不宜建立聚簇索引,特别是随机字符串,会使得系统进行大量的移动操作),且一个表只能有一个聚簇索引。因为由存储引擎实现索引,所以,并不是所有的引擎都支持聚簇索引。目前,只有 solidDB 和 InnoDB 支持。
聚簇索引的结构大致如下:
MySQL 的索引与优化
 

 注:叶子页面包含完整的元组,而内节点页面仅包含索引的列(索引的列为整型)。一些 DBMS 允许用户指定聚簇索引,但是 MySQL 的存储引擎到目前为止都不支持。InnoDB 对主键建立聚簇索引。如果你不指定主键,InnoDB 会用一个具有唯一且非空值的索引来代替。如果不存在这样的索引,InnoDB 会定义一个隐藏的主键,然后对其建立聚簇索引。一般来说,DBMS 都会以聚簇索引的形式来存储实际的数据,它是其它二级索引的基础。

3.1.1、InnoDB 和 MyISAM 的数据布局的比较
为了更加理解聚簇索引和非聚簇索引,或者 primary 索引和 second 索引(MyISAM 不支持聚簇索引),来比较一下 InnoDB 和 MyISAM 的数据布局,对于如下表:

 

CREATE TABLE layout_test (

  col1 int NOT NULL,

  col2 int NOT NULL,

  PRIMARY KEY(col1),

  KEY(col2)

);

 假设主键的值位于 1 —10,000 之间,且按随机顺序插入,然后用 OPTIMIZE TABLE 进行优化。col2 随机赋予 1 —100 之间的值,所以会存在许多重复的值。
(1)   MyISAM 的数据布局
其布局十分简单,MyISAM 按照插入的顺序在磁盘上存储数据,如下:
MySQL 的索引与优化

 注:左边为行号 (row number),从 0 开始。因为元组的大小固定,所以 MyISAM 可以很容易的从表的开始位置找到某一字节的位置。
据些建立的 primary key 的索引结构大致如下:
MySQL 的索引与优化

 注:MyISAM 不支持聚簇索引,索引中每一个叶子节点仅仅包含行号 (row number),且叶子节点按照 col1 的顺序存储。
来看看 col2 的索引结构:

MySQL 的索引与优化

 实际上,在 MyISAM 中,primary key 和其它索引没有什么区别。Primary key 仅仅只是一个叫做 PRIMARY 的唯一,非空的索引而已。

(2)   InnoDB 的数据布局
InnoDB 按聚簇索引的形式存储数据,所以它的数据布局有着很大的不同。它存储表的结构大致如下:

MySQL 的索引与优化

 注:聚簇索引中的每个叶子节点包含 primary key 的值,事务 ID 和回滚指针(rollback pointer)——用于事务和 MVCC,和余下的列(如 col2)。

相对于 MyISAM,二级索引与聚簇索引有很大的不同。InnoDB 的二级索引的叶子包含 primary key 的值,而不是行指针(row pointers),这减小了移动数据或者数据页面分裂时维护二级索引的开销,因为 InnoDB 不需要更新索引的行指针。其结构大致如下:

MySQL 的索引与优化

 聚簇索引和非聚簇索引表的对比:

 MySQL 的索引与优化

 

 3.1.2、按 primary key 的顺序插入行(InnoDB)

如果你用 InnoDB,而且不需要特殊的聚簇索引,一个好的做法就是使用代理主键(surrogate key)——独立于你的应用中的数据。最简单的做法就是使用一个 AUTO_INCREMENT 的列,这会保证记录按照顺序插入,而且能提高使用 primary key 进行连接的查询的性能。应该尽量避免随机的聚簇主键,例如,字符串主键就是一个不好的选择,它使得插入操作变得随机。

 

 3.2、覆盖索引 (Covering Indexes)
如果索引包含满足查询的所有数据,就称为覆盖索引。覆盖索引是一种非常强大的工具,能大大提高查询性能。只需要读取索引而不用读取数据有以下一些优点:
(1)索引项通常比记录要小,所以 MySQL 访问更少的数据;
(2)索引都按值的大小顺序存储,相对于随机访问记录,需要更少的 I /O;
(3)大多数据引擎能更好的缓存索引。比如 MyISAM 只缓存索引。
(4)覆盖索引对于 InnoDB 表尤其有用,因为 InnoDB 使用聚集索引组织数据,如果二级索引中包含查询所需的数据,就不再需要在聚集索引中查找了。
覆盖索引不能是任何索引,只有 B -TREE 索引存储相应的值。而且不同的存储引擎实现覆盖索引的方式都不同,并不是所有存储引擎都支持覆盖索引 (Memory 和 Falcon 就不支持)。
对于索引覆盖查询(index-covered query),使用 EXPLAIN 时,可以在 Extra 一列中看到“Using index”。例如,在 sakila 的 inventory 表中,有一个组合索引(store_id,film_id),对于只需要访问这两列的查询,MySQL 就可以使用索引,如下:

mysql> EXPLAIN SELECT store_id, film_id FROM sakila.inventory\G

*************************** 1. row ***************************

          id: 1

 select_type: SIMPLE

        table: inventory

        type: index

possible_keys: NULL

          key: idx_store_id_film_id

      key_len: 3

          ref: NULL

        rows: 5007

        Extra: Using index

1 row in set (0.17 sec)

在大多数引擎中,只有当查询语句所访问的列是索引的一部分时,索引才会覆盖。但是,InnoDB 不限于此,InnoDB 的二级索引在叶子节点中存储了 primary key 的值。因此,sakila.actor 表使用 InnoDB,而且对于是 last_name 上有索引,所以,索引能覆盖那些访问 actor_id 的查询,如:

 

mysql> EXPLAIN SELECT actor_id, last_name

    -> FROM sakila.actor WHERE last_name = ‘HOPPER’\G

*************************** 1. row ***************************

          id: 1

 select_type: SIMPLE

        table: actor

        type: ref

possible_keys: idx_actor_last_name

          key: idx_actor_last_name

      key_len: 137

          ref: const

        rows: 2

        Extra: Using where; Using index

 

3.3、利用索引进行排序
MySQL 中,有两种方式生成有序结果集:一是使用 filesort,二是按索引顺序扫描。利用索引进行排序操作是非常快的,而且可以利用同一索引同时进行查找和排序操作。当索引的顺序与 ORDER BY 中的列顺序相同且所有的列是同一方向(全部升序或者全部降序) 时,可以使用索引来排序。如果查询是连接多个表,仅当 ORDER BY 中的所有列都是第一个表的列时才会使用索引。其它情况都会使用 filesort。

create table actor(

actor_id int unsigned NOT NULL AUTO_INCREMENT,

name      varchar(16) NOT NULL DEFAULT ”,

password        varchar(16) NOT NULL DEFAULT ”,

PRIMARY KEY(actor_id),

 KEY    (name)

) ENGINE=InnoDB

insert into actor(name,password) values(‘cat01′,’1234567’);

insert into actor(name,password) values(‘cat02′,’1234567’);

insert into actor(name,password) values(‘ddddd’,’1234567′);

insert into actor(name,password) values(‘aaaaa’,’1234567′);

 

 

mysql> explain select actor_id from actor order by actor_id \G

*************************** 1. row ***************************

          id: 1

 select_type: SIMPLE

        table: actor

        type: index

possible_keys: NULL

          key: PRIMARY

      key_len: 4

          ref: NULL

        rows: 4

        Extra: Using index

1 row in set (0.00 sec)

 

mysql> explain select actor_id from actor order by password \G

*************************** 1. row ***************************

          id: 1

 select_type: SIMPLE

        table: actor

        type: ALL

possible_keys: NULL

          key: NULL

      key_len: NULL

          ref: NULL

        rows: 4

        Extra: Using filesort

1 row in set (0.00 sec)

 

mysql> explain select actor_id from actor order by name \G

*************************** 1. row ***************************

          id: 1

 select_type: SIMPLE

        table: actor

        type: index

possible_keys: NULL

          key: name

      key_len: 18

          ref: NULL

        rows: 4

        Extra: Using index

1 row in set (0.00 sec)

 当 MySQL 不能使用索引进行排序时,就会利用自己的排序算法 (快速排序算法) 在内存 (sort buffer) 中对数据进行排序,如果内存装载不下,它会将磁盘上的数据进行分块,再对各个数据块进行排序,然后将各个块合并成有序的结果集(实际上就是外排序)。对于 filesort,MySQL 有两种排序算法。
(1)两遍扫描算法 (Two passes)
实现方式是先将须要排序的字段和可以直接定位到相关行数据的指针信息取出,然后在设定的内存(通过参数 sort_buffer_size 设定)中进行排序,完成排序之后再次通过行指针信息取出所需的 Columns。
注:该算法是 4.1 之前采用的算法,它需要两次访问数据,尤其是第二次读取操作会导致大量的随机 I / O 操作。另一方面,内存开销较小。
(3)   一次扫描算法 (single pass)
该算法一次性将所需的 Columns 全部取出,在内存中排序后直接将结果输出。
注:从 MySQL 4.1 版本开始使用该算法。它减少了 I / O 的次数,效率较高,但是内存开销也较大。如果我们将并不需要的 Columns 也取出来,就会极大地浪费排序过程所需要的内存。在 MySQL 4.1 之后的版本中,可以通过设置 max_length_for_sort_data 参数来控制 MySQL 选择第一种排序算法还是第二种。当取出的所有大字段总大小大于 max_length_for_sort_data 的设置时,MySQL 就会选择使用第一种排序算法,反之,则会选择第二种。为了尽可能地提高排序性能,我们自然更希望使用第二种排序算法,所以在 Query 中仅仅取出需要的 Columns 是非常有必要的。

当对连接操作进行排序时,如果 ORDER BY 仅仅引用第一个表的列,MySQL 对该表进行 filesort 操作,然后进行连接处理,此时,EXPLAIN 输出“Using filesort”;否则,MySQL 必须将查询的结果集生成一个临时表,在连接完成之后进行 filesort 操作,此时,EXPLAIN 输出“Using temporary;Using filesort”。

 

3.4、索引与加锁
索引对于 InnoDB 非常重要,因为它可以让查询锁更少的元组。这点十分重要,因为 MySQL 5.0 中,InnoDB 直到事务提交时才会解锁。有两个方面的原因:首先,即使 InnoDB 行级锁的开销非常高效,内存开销也较小,但不管怎么样,还是存在开销。其次,对不需要的元组的加锁,会增加锁的开销,降低并发性。
InnoDB 仅对需要访问的元组加锁,而索引能够减少 InnoDB 访问的元组数。但是,只有在存储引擎层过滤掉那些不需要的数据才能达到这种目的。一旦索引不允许 InnoDB 那样做(即达不到过滤的目的),MySQL 服务器只能对 InnoDB 返回的数据进行 WHERE 操作,此时,已经无法避免对那些元组加锁了:InnoDB 已经锁住那些元组,服务器无法解锁了。
来看个例子:

create table actor(

actor_id int unsigned NOT NULL AUTO_INCREMENT,

name      varchar(16) NOT NULL DEFAULT ”,

password        varchar(16) NOT NULL DEFAULT ”,

PRIMARY KEY(actor_id),

 KEY    (name)

) ENGINE=InnoDB

insert into actor(name,password) values(‘cat01′,’1234567’);

insert into actor(name,password) values(‘cat02′,’1234567’);

insert into actor(name,password) values(‘ddddd’,’1234567′);

insert into actor(name,password) values(‘aaaaa’,’1234567′);

SET AUTOCOMMIT=0;

BEGIN;

SELECT actor_id FROM actor WHERE actor_id < 4

AND actor_id <> 1 FOR UPDATE;

 该查询仅仅返回 2 —3 的数据,实际已经对 1 —3 的数据加上排它锁了。InnoDB 锁住元组 1 是因为 MySQL 的查询计划仅使用索引进行范围查询(而没有进行过滤操作,WHERE 中第二个条件已经无法使用索引了):

 

mysql> EXPLAIN SELECT actor_id FROM test.actor

    -> WHERE actor_id < 4 AND actor_id <> 1 FOR UPDATE \G

*************************** 1. row ***************************

          id: 1

 select_type: SIMPLE

        table: actor

        type: index

possible_keys: PRIMARY

          key: PRIMARY

      key_len: 4

          ref: NULL

        rows: 4

        Extra: Using where; Using index

1 row in set (0.00 sec)

 

mysql>

 表明存储引擎从索引的起始处开始,获取所有的行,直到 actor_id<4 为假,服务器无法告诉 InnoDB 去掉元组 1。
为了证明 row 1 已经被锁住,我们另外建一个连接,执行如下操作:

SET AUTOCOMMIT=0;

BEGIN;

SELECT actor_id FROM actor WHERE actor_id = 1 FOR UPDATE;

该查询会被挂起,直到第一个连接的事务提交释放锁时,才会执行(这种行为对于基于语句的复制 (statement-based replication) 是必要的)。
如上所示,当使用索引时,InnoDB 会锁住它不需要的元组。更糟糕的是,如果查询不能使用索引,MySQL 会进行全表扫描,并锁住每一个元组,不管是否真正需要。

本文永久更新链接地址:http://www.linuxidc.com/Linux/2017-02/140304.htm

正文完
星哥玩云-微信公众号
post-qrcode
 0
星锅
版权声明:本站原创文章,由 星锅 于2022-01-22发表,共计9993字。
转载说明:除特殊说明外本站文章皆由CC-4.0协议发布,转载请注明出处。
【腾讯云】推广者专属福利,新客户无门槛领取总价值高达2860元代金券,每种代金券限量500张,先到先得。
阿里云-最新活动爆款每日限量供应
评论(没有评论)
验证码
【腾讯云】云服务器、云数据库、COS、CDN、短信等云产品特惠热卖中