首页 > 数据库 >MySQL之索引

MySQL之索引

时间:2024-03-04 21:25:33浏览次数:21  
标签:index name 查询 索引 MySQL 数据 节点

索引

知识回顾:数据都是存在于硬盘上的,查询数据不可避免的需要进行IO操作

索引:就是一种数据结构,类似于书的目录。意味着以后在查询数据的时候应该先找目录再找数据,而不是一页一页的番薯,从而提升查询熟读降低IO操作。

索引在MySQL中也叫“键”,是存储引擎用于快速查找记录的一种数据结构

  • primary key
  • unique
  • index key

注意 foreign key 不是用来加速查询用的,不在我们的研究范围之内

上面的三种key,前面的两种除了可以增加查询速度之外各自还具有约束条件,而最后一种index key没有任何的约束条件,只是用来帮助你快速查询数据。

本质

通过不断的缩小想要的数据范围筛选出最终的结果,同时将随机事件(一页一页的翻)变成顺序事件(先找目录,再找数据),也就是说有了索引机制,我们可以总是用一种固定的方式查找数据。

一张表中可以有多个索引(多个目录)

索引虽然能够帮助你加快查询速度但是也有缺点

缺点:

  1. 当表中有大量数据存在的前提下,创建索引速度回很慢
  2. 在索引创建完毕之后,对表的查询性能会大幅度提升,但是写的性能也会大幅度的降低。

总结:

​ 索引不要随意的创建!!!

索引操作

查看索引

语法

show index from 表名;

示例

 show index from user_info;
+-----------+------------+----------+--------------+-------------+-----------+-------------+----------+--------+------+------------+---------+---------------+
| Table     | Non_unique | Key_name | Seq_in_index | Column_name | Collation | Cardinality | Sub_part | Packed | Null | Index_type | Comment | Index_comment |
+-----------+------------+----------+--------------+-------------+-----------+-------------+----------+--------+------+------------+---------+---------------+
| user_info |          0 | PRIMARY  |            1 | id          | A         |           2 |     NULL | NULL   |      | BTREE      |         |               |
+-----------+------------+----------+--------------+-------------+-----------+-------------+----------+--------+------+------------+---------+---------------+

desc user_info;
+----------+--------------+------+-----+---------+----------------+
| Field    | Type         | Null | Key | Default | Extra          |
+----------+--------------+------+-----+---------+----------------+
| id       | int(11)      | NO   | PRI | NULL    | auto_increment |
| name     | varchar(255) | YES  |     | NULL    |                |
| password | varchar(255) | YES  |     | NULL    |                |
+----------+--------------+------+-----+---------+----------------+

创建索引

对于非主键、非唯一约束、非外键的字段,可以创建普通索引

语法

create index  索引名 on 表名(字段名);
# 索引名的命名规则一般是:index_表名_列名

示例

create index index_name on user_info(name);
Query OK, 0 rows affected (0.04 sec)
Records: 0  Duplicates: 0  Warnings: 0

删除索引

语法

drop index 索引名 on 表名

示例

drop index index_name on user_info;
Query OK, 0 rows affected (0.01 sec)
Records: 0  Duplicates: 0  Warnings: 0

索引的数据结构

MySQL的索引是在存储引擎层实现的,不同的存储引擎有不同的结构,主要包含以下几种:

索引结构 描述
B+Tree索引 最常见的索引类型,大部分引擎都支持B+树索引。
Hash索引 底层数据结构是用哈希表实现的,只有精确匹配索引列的查询才有效,不支持范围查询。
R-tree(空间索引) 空间索引是MyISAM引擎的一个特殊索引类型,主要用于地理空间数据类型,通常使用较少
Full-text(全文索引) 是一种通过建立倒排索引,快速匹配文档的方式。类似于Lucene,Solr,ES。

B+Tree索引

MySQL索引数据结构对经典的B+Tree进行了优化。在原B+Tree的基础上,增加一个指向相邻叶子节点的链表指针,就形成了带有顺序指针的B+Tree,提高区间访问的性能。

只有叶子结点存放的是真实的数据,其他节点存放的是虚拟数据,仅仅是用来指路的。树的层级越高查询数据所需要经历的步骤就越多。而一个磁盘块存储是有限制的。

image

Hash索引

哈希索引就是采用一定的hash算法,将键值换算成新的hash值,映射到对应的槽位上,然后存储在hash表中。

如果两个(或多个)键值,映射到一个相同的槽位上,他们就产生了hash冲突(也称为hash碰撞),可以通过链表来解决。

image

Hash索引特点:

  1. Hash索引只能用于对等比较(=,in),不支持范围查询(between,>,<,....)
  2. 无法利用索引完成排序操作
  3. 查询效率高,通常只需要一次检索就可以了,效率通常要高于B+tree索引

存储引擎支持:

​ 在MsaL中,支持hash索引的是Memory引擎,而innoDB中具有自适应hash功能,hash索引是存储引擎根据B+Tree索引在指定条件下自动构建的。

思考题:

1.为什么InnoDB存储引擎选择使用B+Tree索引结构?(面试题)

  • 相对于二叉树,层级更少,搜索效率越高。
  • 对于B-tree,无论是叶子节点还是非叶子节点,都会保存数据,这样导致一页中存储的键值减少,指针跟着减少,要同样保存大量数据,只能增加树的高度,导致性能降低。
  • 相对Hash索引,B+Tree支持范围匹配以及排序操作。

2.为什么建议你将id字段作为索引?

  • 占的空间少,一个磁盘块能够存储的数据多
  • 那么就降低了树的高度,从而减少了查询次数

聚集索引(主键索引)

将数据存储与索引放到了一块,索引结构的叶子节点保存了行数据(所有字段的值)。

  • 聚集索引指的就是主键
  • Innodb 只有两个文件,直接将主键存放在了idb表中
  • MyIsam 三个文件,单独将索引存在一个文件

InnoDB的聚集索引图解

image

辅助索引(非聚集索引)

将数据与索引分开存储,索引结构的叶子节点指向了数据对应的位置

查询数据的时候不可能一直使用到主键,也有可能会用到name、password等字段,那么这个时候就没有办法利用聚集索引,所以我们就可以根据情况给其他字段设置辅助索引(也是一个B+树)

  • 叶子结点存放的是数据对应的主键值
  • 先按照堵住索引拿到数据的主键值,之后还是需要主键的聚集索引里面来查询数据。
# 给name设置辅助索引(按照用户的名字来做辅助索引)
select name from user where name='xiao';

InnoDB的非聚集索引图解

image

图示根据主键查询的过程:

image

整个查询的过程如下:

  1. 查询 id(主键) 为 18 的数据,SELECT id, name, age WHERE id = 18。

  2. 首先在「根节点:节点一」上,id = 18 落在了 15 <= id < 56 范围之内,这样我们就知道了下级节点「非叶子节点:节点2-1」的地址。

  3. 根据【步骤2】得到的「非叶子节点:节点2-1」的地址,找到对应的「非叶子节点:节点2-1」。然后,id = 18 又落在了 15 <= id < 20 范围之内,这样我们就知道了再下一级节点「叶子节点:节点3-1」的地址。

  4. 根据【步骤3】得到的「叶子节点:节点3-1」的地址,找到对应的「叶子节点:节点3-1」。最后,在「叶子节点:节点3-1」这个节点上找到 id = 18 对应的数据 {“id”: 18, “name”: “King”, “age”: 1

覆盖索引

覆盖索引(covering index ,或称为索引覆盖)即从非主键索引中就能查到的记录,而不需要查询主键索引中的记录,避免了回表的产生减少了树的搜索次数,显著提升性能。 如果一个索引包含了(或覆盖了)满足查询语句中字段与条件的数据就叫 做覆盖索引。

  • 在辅助索引的叶子结点就已经拿到了需要的数据
例如:
# 给name设置辅助索引(按照用户的名字来做辅助索引)
select name from user where name='xiao';
这种情况已经可以使用辅助索引来直接找到对应的数据了,就不需要再使用主键值来查找了,这就叫覆盖索引
select age from user where name='xiao';
这种情况使用辅助索引来找对应的数据了,但是我们并没有得到相对应的数据,这就叫非覆盖索引

唯一索引(unique)

  • 唯一索引是指该索引的所有值都是唯一的,不允许出现重复值。

语法

  • MySQL中可以通过以下语法创建唯一索引
ALTER TABLE table_name ADD UNIQUE KEY index_name (column);

注意事项

  • 与普通索引不同的是,如果尝试向包含唯一索引的列中插入重复的值,则会引发错误。
  • 唯一索引可以用于确保数据的一致性和完整性,并且可以帮助提高查询性能。

组合索引(联合索引)

  • 组合索引是一种由两个或更多列组成的索引。
  • 当查询涉及多个列时,组合索引可以大大提高查询性能。

语法

  • MySQL中可以通过以下语法创建组合索引
ALTER TABLE table_name ADD INDEX index_name (column1, column2, ...);
  • 需要注意的是,组合索引中的列顺序很重要。
  • 在执行查询时,MySQL会首先使用最左的列进行匹配,然后逐步向右扩展。
  • 因此,应该将最常用的查询条件放在组合索引的前面。

缺点

  • 例如,如果添加了不需要的列,或者删除了不需要的列,可能会导致组合索引变得无效。
  • 此外,如果创建了太多的组合索引,也可能会增加索引维护的成本。

全文索引

  • 全文索引是一种特殊的索引,它可以用来存储和检索文本数据。
  • 全文索引可以包含单词、短语和其他类型的文本内容,并支持模糊匹配和近似匹配。

语法

  • MySQL中可以通过以下语法创建全文索引
CREATE FULLTEXT INDEX index_name ON table_name (column);

注意事项

  • 需要注意的是,只有MyISAM和InnoDB存储引擎支持全文索引。
  • 此外,创建全文索引可能会增加索引维护的成本,并且可能会降低其他类型的查询性能。
  • 因此,在创建全文索引时需要权衡其利弊。

前缀索引

  • 前缀索引是一种特殊的索引,它只存储索引列的一部分,而不是完整的值。
  • 前缀索引通常用于处理非常大的列,例如IP地址或邮政编码。
  • 在这种情况下,全列索引可能会消耗大量的存储空间,并且可能会导致查询性能下降。
  • 前缀索引可以大大减少索引大小,并且可以更快地执行范围查询。

语法

  • MySQL中可以通过以下语法创建前缀索引
ALTER TABLE table_name ADD INDEX index_name (column(length));
  • 其中,length参数表示要保留的字符数。

示例

  • 例如,如果我们有一个邮政编码列,并且我们只需要后两位数字进行索引,我们可以创建以下前缀索引:
ALTER TABLE table_name ADD INDEX idx_postcode(postcode(2));
  • 在这个例子中,索引只会包含邮政编码的最后两位数字。
  • 当我们执行范围查询时,MySQL可以直接使用前缀索引来缩小搜索范围,从而提高查询性能。

标签:index,name,查询,索引,MySQL,数据,节点
From: https://www.cnblogs.com/xiao01/p/18052700

相关文章

  • MySQL-18 MySQL8其他新特性
    C-18.MySQL8其他新特性1.MySQL8新特性概述MySQL从5.7版本直接跳跃发布了8.0版本,可见是一个令人兴奋的里程碑的版本。MySQL8版本在功能上,做了显著的改进与增强,开发者对MySQL的源代码进行了重构,最突出的一点是对MySQLOptimizer优化器进行了改进。不仅在速度上得到了改善,还为用......
  • C++ mySQL数据库连接池(windows平台)
    C++MySQL数据库连接池新手学了C++多线程,看了些资料练手写了C++数据库连接池小项目,自己的源码地址关键技术点MySQL数据库编程、单例模式、queue队列容器、C++11多线程编程、线程互斥、线程同步通信和unique_lock、基于CAS的原子整形、智能指针shared_ptr、lambda表达式、生产......
  • 【Redis】Redis如何保证和MySQL数据库的数据一致性
    保障MySQL和Redis数据一致性需要使用不同的策略和技术,因为两者是不同的数据存储系统。以下是一些常见的方法:1.数据同步MySQL数据同步至Redis使用事件驱动机制:当MySQL中的数据更新时,通过触发器或者其他事件驱动的机制,将数据同步至Redis。定时任务:定期轮询MySQL数据......
  • mysql语句的执行顺序
    一、sql执行的流程1mysql客户端发送查询请求到服务器。2mysql服务器接收请求并处理,mysql解析器解析查询语句,进行语法分析,确保查询语句符合mysql的语法要求。3mysql的查询优化器对sql语句进行优化处理(选择最合适的索引,或使用其它技术来提高性能),生成执行计划。4mysql执行引......
  • Ubuntu安装zabbix,初始化数据库报没有这个文件 /usr/share/doc/zabbix-sql-scripts/mys
    报错信息如下: 解决方法:1、先查看是否安装了zabbix-server-mysql,我这里是已经安装过了,但是初始化还是报错找不到文件 2、去zabbix下载对应版本的源码,然后进行手动安装下载链接:https://www.zabbix.com/download_sources#60LTS 3、源码下载后解压,在database文件中找到m......
  • 记一次在oracle数据库中添加全文索引的过程
    前景:在创建全文索引之前,有如下几个内容需要确认下:1.全文索引不支持nvarchar2类型,所以需要创建全文索引的字段的字段类型,如果是这个类型,可以修改为varchar2();2.如果需要创建全文索引的字段包含(1、中文、英文结合,或者纯英文;2、中文、数字结合,或者纯数字的)这几种情况的,需要对......
  • Mysql基本语法笔记
    DDL--操作数据库1.查询SHOWDATABASES;2.创建CREATEDATABASE数据库名称CHARACTERSETutf8;如果不存在创建CREATEDATABASEIFNOTEXISTS数据库名称;3.删除DROPDATABASE数据库名称;如果存在删除DROPDATABASEIFEXISTS数据库名称;4.使用数据库查看当前数......
  • docker环境部署容器之间互通:Hyperf+MySQL
    在docker中要使Hyperf和Mysql网络互通,则需要在创建相应容器的时候指定共享网络,步骤如下:一、首先我们指定共享网络dockernetworkcreate--subnet172.18.0.1/16test  //整条命令复制执行就行不用改这个ip二、创建hyperf命令dockerrun--namehyperf-vD:\hyperf202......
  • MySQL index and dead lock
    MySQL        Analyze:step1.Thread(test1)use"selectforupdate"lockcommonageindexinrange(10,30]step 2.Thread(test2)use"selectforupdate"lockcommonageindexinrange(40,60]step 3.Thread(test2)trytoi......
  • mysqldump: Error: Binlogging on server not active
    1.问题今天在备份数据库时遇到问题,提示二进制日志尚未开启,无法使用--master-data=2选项2.解决启用二进制日志记录,打开MySQL配置文件(通常是my.cnf或my.ini),添加以下选项即可。保存并重新启动MySQL服务器#开启二进制日志binloglog-bin=mysql-bin#开启Binlog一般只需......