美文网首页
一个不留神,索引就创建重复了

一个不留神,索引就创建重复了

作者: _江南一点雨 | 来源:发表于2022-03-08 15:54 被阅读0次

相信没有人会故意创建重复的冗余的索引,很多重复和冗余的索引都是在不经意间创建的,今天松哥来和大家捋一捋这个问题。

因为我们日常在使用 MySQL 的过程中,基本上都是使用 InnoDB 引擎,所以接下来的讨论主要是基于 InnoDB 引擎的 B+Tree 索引来讨论,其他的哈希索引全文索引等不在讨论范围种。

1. 与联合索引重复

在前面的文章中,松哥通过好几篇文章和大家分享了联合索引,包括它涉及到的覆盖索引、前缀匹配等等,联合索引好用,但是对联合索引理解不到位的话,可能会创建出如下的重复索引:

CREATE TABLE `user2` (
  `id` int(11) unsigned NOT NULL AUTO_INCREMENT,
  `username` varchar(32) COLLATE utf8mb4_unicode_ci DEFAULT NULL,
  `address` varchar(32) COLLATE utf8mb4_unicode_ci DEFAULT NULL,
  `password` varchar(32) COLLATE utf8mb4_unicode_ci DEFAULT NULL,
  `email` varchar(16) COLLATE utf8mb4_unicode_ci DEFAULT NULL,
  PRIMARY KEY (`id`),
  KEY `user_index1` (`username`,`address`),
  KEY `user_index2` (`username`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;

可以看到,这里创建了两个索引:

  • user_index1:这个索引包含两个字段,username 在前 address 在后。
  • user_index2:这个索引包含一个字段 username。

其实 MySQL 中的 like 关键字也能用索引!一文中,松哥跟大家聊了索引的最左匹配原则,即:

(username,address) 索引既可以当成联合索引来用,也可以通过最左匹配原则当成单独的 (username) 索引来用。

所以,如果再为 username 字段单独创建一个索引就没有必要了,这反而会导致增删改的时候速度变慢。

不过怎么说呢,上面这个结论适用于 99% 的场景,可能会有一些特殊情况,例如想把 (username) 和某一个特别长的字段建立一个联合索引,此时如果单独使用 username 字段进行搜索的话,效率可能降低,此时视搜索的重要程度,看是否需要创建一个重复的索引。

2. 主键加入联合索引中

来看看下面这个索引:

CREATE TABLE `user2` (
  `id` int(11) unsigned NOT NULL AUTO_INCREMENT,
  `username` varchar(32) COLLATE utf8mb4_unicode_ci DEFAULT NULL,
  `address` varchar(32) COLLATE utf8mb4_unicode_ci DEFAULT NULL,
  `password` varchar(32) COLLATE utf8mb4_unicode_ci DEFAULT NULL,
  `email` varchar(16) COLLATE utf8mb4_unicode_ci DEFAULT NULL,
  PRIMARY KEY (`id`),
  KEY `user_index` (`username`,`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;

一个名为 user_index 的索引中包含了两个字段 username 和 id,其中 id 是主键。

什么是 MySQL 的“回表”?一文中,松哥和大家聊了,索引按照物理存储方式可以分为聚簇索引和非聚簇索引。

我们日常所说的主键索引,其实就是聚簇索引(Clustered Index);主键索引之外,其他的都称之为非主键索引,非主键索引也被称为二级索引(Secondary Index),或者叫作辅助索引。

对于主键索引和非主键索引,使用的数据结构都是 B+Tree,唯一的区别在于叶子结点中存储的内容不同:

  • 主键索引的叶子结点存储的是一行完整的数据。
  • 非主键索引的叶子结点存储的则是主键值以及索引列的值。

这是两者最大的区别。

既然主键已经存在于叶子结点中,那当然没有在联合索引中加入主键了。

好啦,几个小小的注意点,希望能给小伙伴们启发。

参考资料:

  • 《高性能 MySQL》

相关文章

  • 一个不留神,索引就创建重复了

    相信没有人会故意创建重复的冗余的索引,很多重复和冗余的索引都是在不经意间创建的,今天松哥来和大家捋一捋这个问题。 ...

  • Neo4j-spring data 基本使用,第三发

    创建索引 CREATE INDEX ON :Customer (userId); 创建索引 用一个属性值判断节点是否重复

  • elasticsearch 的 python API

    导入 es 创建索引 其中的 acknowledged 字段表示创建操作执行成功 重复创建索引,会引发 400 错...

  • elasticsearch简单操作

    通过postman操作elasticsearch版本为7.8.0 添加索引 不能重复的创建索引 查看索引 字段的含...

  • 51-MySQL-索引的设计原则

    一、高效索引 为了使索引的使用效率更高,在创建索引时,必须考虑在哪些字段上创建索引和创建什么类型的索引。索引设计不...

  • ElasticSearch学习笔记5--索引

    5.1 创建索引 (1)简单方式 响应 (2)索引名不能包含大写字母 (3)重复创建 (4)指定参数 响应 5.2...

  • Lucene 6.6.1源码分析---IndexWriter创建

    索引的创建 Lucene创建索引的步骤非常简单,只需要创建一个IndexWriter实例即可,这个实例就代表一个索...

  • Python中的集合(Set)

    在Python中,集合(Set) 是一个无序、不重复的序列,它不支持索引。 创建集合 一般在创建集合时,分为创建空...

  • MySQL索引

    索引创建时,要考虑到字段的重复性,重复越少的字段,索引查询的速度就越快, 2,对于多个and连接的查询条件,可以考...

  • postgresql的常用查询sql(20.6.2)

    查看表的大小 查询表与索引的关系信息 查看索引的大小 查看索引使用情况 查看唯一性索引 查询重复创建的索引 查看执...

网友评论

      本文标题:一个不留神,索引就创建重复了

      本文链接:https://www.haomeiwen.com/subject/kfmarrtx.html