MySQL索引优化实战宝典

news2025/1/11 22:48:58

        MySQL索引是MySQL数据库用于快速查找和访问数据的一种数据结构,它就像书的目录一样,可以帮助数据库系统更快地定位到所需数据的位置,从而大大提高查询性能。

        下面来看一下索引分类

一、索引分类        

        本文介绍平时使用最多的基于 InnnoDB 存储引擎的索引,InnoDB 的索引是基于 B+ 树的。B+ 树是一种自平衡的树形数据结构,所有数据都存储在叶子节点上,并且叶子节点之间通过双向链表连接,形成有序序列,非常适合范围查询和全表扫描。

                

        索引的具体分类如下:

        聚集索引

        表中数据按照主键顺序,同时叶子节点存放的为整张表的行记录数据,也将叶子节点称为数据页,对于主键的排序和范围查找速度非常快。

        辅助索引

        叶子节点并不包含行记录的全部数据。每个叶子节点除了包含键值外,还包含一个书签,该书签用来告诉 Innodb 存储引擎哪里可以找到与索引对应的行数据。由于 Innodb 存储引擎表是索引组织表,因此 Innodb 存储引擎的辅助索引书签就是相应行记录的聚集索引键。当通过辅助索引来寻找数据时,Innodb 存储引擎会遍历辅助索引并通过叶级别的指针获得指向主键索引的主键,然后在通过主键索引来找到一个完成的行记录(回表)。

        联合索引

        对多个列同时创建的索引,它基于这些列的值来组织数据,优化查询性能。联合索引主要应用于涉及多个列的查询条件,可以显著加快数据检索速度,减少磁盘 I/O。

        覆盖索引

        从辅助索引就能查到想要的记录,而不需要回表。

        前缀索引

        用字段的一部分做索性,例如有邮箱字段 email,建立索引时可以使用 email(6) 指定只使用前六个字符作为索引。使用前缀索引可以极大的节省空间,但是使用前缀索引后不能使用覆盖索引了,因为引擎不知道前缀索引是否包含整个字符串,所以需要额外的回表操作才能判断。

        索引应尽可能的小一些,因为这样占用的空间小,相同的数据页就能容纳更多的索引数据。

二、普通索引和唯一索引如何选择

        分别从查询过程和更新过程来说明一下如何选择普通索引和唯一索引。

        查询过程

        假设执行的语句为 select id from T where k=5。这个查询语句的查找过程,先是通过 B+ 树从树根开始,按层搜索到叶子节点,然后可以认为数据页内部通过二分法来定位记录。

  • 对于普通索引来说,查找到满足条件的第一个记录后,需要查找下一个记录,直到碰到第一个不满足 k=5 条件的记录。如果这条数据在当前数据页的最后,就需要读下一个页,但一个页可以放近千个 key,这种情况出现的概率很低;
  • 对于唯一索引来说,由于唯一索引的唯一性,查找到第一个满足条件的记录后,就会停止继续检索。

        这个不同带来的性能差距微乎其微。

        更新过程

        当需要更新一个数据页时,如果数据页在内存中就直接更新,而如果这个数据页还没有在内存中的话,在不影响数据一致性的前提下,InnoDB 会将这些更新操作缓存在 change buffer 中,这样就不需要从磁盘中读入这个数据页了。在下次查询需要访问这个数据页的时候,将数据页读入内存,然后执行 change buffer 中与这个页有关的操作。通过这种方式就能保证这个数据逻辑的正确性。

        将 change buffer 中的操作应用到原数据页,得到最新结果的过程称为 merge。除了访问这个数据页会触发 merge 外,系统有后台线程会定期 merge。在数据库正常关闭(shutdown)的过程中,也会执行 merge 操作。

        对于唯一索引来说,所有的更新操作都要先判断这个操作是否违反唯一性约束。比如,要插入 (4,400) 这个记录,就要先判断现在表中是否已经存在 k=4 的记录,而这必须要将数据页读入内存才能判断。如果都已经读入到内存了,那直接更新内存会更快,就没必要使用 change buffer 了。

        因此,唯一索引的更新就不能使用 change buffer,实际上也只有普通索引可以使用。

        普通索引和唯一索引应该怎么选择。其实,这两类索引在查询能力上是没差别的,主要考虑的是对更新性能的影响。所以,建议尽量选择普通索引。

三、索引优劣判断

        通过 show index from table 可以观察表上的索引,其中有个属性相当重要——Cardinality,这个值非常关键。它是评估索引质量与效率的一个重要因素。在MySQL等数据库系统中,索引基数越高,意味着索引列的唯一性越强,索引筛选数据的能力也就越强。

        优化器会根据这个值来判断是否使用这个索引。但是这个值并不是实时更新的,因为代价太大。可以通过 analyze table 来更新索引。

        Cardinality/rows 应尽可能接近 1,如果非常小,那么就可以考虑有没有必要创建这个索引。

四、索引优化

        4.1 MRR优化        

        MySQL 5.6 版本开始支持 Multi-Range Read 优化。目的是为了减少磁盘的随机访问,并且将随机访问转化为较为顺序的数据访问。原理如下:

        范围扫描与回表: 当查询涉及二级索引,并且需要根据索引结果回表查询其他非索引列时,如果没有 MRR 优化,MySQL 会先扫描二级索引获取所有符合条件的记录的主键值,然后逐个通过主键值回表查询对应的行数据。这个过程中的回表操作往往会造成大量的随机 I/O。

        MRR 优化步骤:

  • 收集排序: MRR 优化在扫描二级索引的过程中,会收集所有需要回表查询的主键值,并对这些主键值进行排序。
  • 批量回表: 排序后的主键值集合按照顺序进行批量回表查询,即一次性请求一批相邻主键值对应的行数据,而不是一次一个随机访问。
  • 减少随机 I/O: 通过批量读取和顺序访问,MRR 优化有效地减少了在硬盘上寻找数据的随机I/O 次数,转而采用更高效的顺序 I/O,这在大量数据处理时尤其重要。

        4.2 索引下推优化

        Index Condition Pushdown 同样是 MySQL 5.6 开始支持的一种根据索引进行查询的优化方式。当进行索引查询时首先根据索引来查找记录,然后在根据 where 条件来过滤记录。在支持索引下推后,MySQL 数据会在去除索引的同时,判断是否可以进行 where 条件过滤,也就是将where 的部分过滤操作放在了存储引擎层,减少了回表操作,从而提高了性能。

  • 在不使用 ICP 的情况下,在使用非主键索引(又叫普通索引或者二级索引)进行查询时,存储引擎通过索引检索到数据,然后返回给 MySQL 服务器,服务器然后判断数据是否符合条件 。
  • 在使用 ICP 的情况下,如果存在某些被索引的列的判断条件时,MySQL 服务器将这一部分判断条件传递给存储引擎,然后由存储引擎通过判断索引是否符合 MySQL 服务器传递的条件,只有当索引符合条件时才会将数据检索出来返回给 MySQL 服务器 。
  • 索引条件下推优化可以减少存储引擎查询基础表的次数,也可以减少 MySQL 服务器从存储引擎接收数据的次数。

        注意:MySQL 的索引下推(Index Condition Pushdown, ICP)优化特性只在查询条件包含索引列,并且这部分条件能够利用索引进行过滤的情况下生效。

        4.3 最左匹配原则

        如果我们使用的是复合索引,应该尽量遵循最左前缀匹配原则。MySQL 会一直向右匹配直到遇到范围查询(>、<、between、like)就停止匹配。

select * from t where a = 1 and b = 2 and c > 3 and d = 4;

        如上 sql,那我们建立的复合索引应该是index(a,b,d,c)而不是index(a,b,c,d),因为 c 是范围查询,当 MySQL 遇到范围查询就停止索引匹配。

        还要注意一点:最左前缀原则不但是复合索引的最左N个字段;也可以是单列(字符串类型)索引的最左M个字符。

        例如我们常说的like关键字,尽量不要使用全模糊查询,因为这样用不到索引,所以建议使用右模糊查询。

        4.3 覆盖索引

        很多时候还可以用覆盖索引来优化 sql。

        情况一:sql只查询主键作为返回值。

        主键索引(聚簇索引)的叶子节点是整行数据,而普通索引(二级索引)的叶子节点是主键的值。所以当我们的sql值查询主键值,可以直接获取对应叶子节点的内容,而避免了回表。

        情况二:sql查询字段就在索引里

        复合索引,假如我们有一个复合索引 index(name,age),sql 如下

select name, age from t where name like '苏%';

        由于字段 name 是右模糊查询所以可以走复合索引,然后匹配到 name 时,不需要回表,因为 sql 只查询 name 和 age,索引直接就返回索引值就 ok 了。

        4.4 order by 索引优化

        order by 后面的字段尽量是带索引的,这样能避免 sort_buffer 进行排序。假如一条 sql,根据生日查询所有学生的信息:

select * from student order by birthday desc;

        那么为了提升查询性能,应该在birthday字段上建立索引。

        select 后面不要带上不必要的字段,因为如果当行长度太长会导致查询数据太多,MySQL 会利用 rowid 排序来代替全字段排序,这样会导致回表操作。查什么字段就跟上什么字段。

        如果 MySQL 认为内存足够大,会优先选择全字段排序,把需要的字段都放到 sort_buffer中, 这样排序后就会直接从内存里面返回查询结果了,不用再回到原表去取数据。

        4.5 or 索引优化

        在 InnoDB 引擎下or关键字无法使用复合索引。

select id, product_name from orders where mobile = '12345678' or user_id = 6;

        一般我们为了提升上面 sql 的效率,会想着为字段 mobile 和 user_id 建立一个复合索引index(mobile, user_id); 可是我们看执行计划的话提示并没有使用复合索引,所以 or 关键字无法命中复合索引。        

往期经典推荐

深入JVM内核揭示Java多态背后的神秘机制-CSDN博客

MySQL数据更新流程原来这么复杂-CSDN博客

MySQL自增主键有什么作用?来自大厂的使用经验_数据库主键自增-CSDN博客

MySQL计数优化探秘:COUNT(*)、COUNT(主键)与索引字段,谁是性能王者?-CSDN博客

TiDB内核解密:揭秘其底层KV存储引擎如何玩转键值对_tidb 的key value是如何做到的-CSDN博客

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1548465.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

盘点4款最适合教学的电路仿真软件

在现代电子教育中&#xff0c;电路仿真软件扮演着至关重要的角色。它们不仅能够帮助学生更好地理解电子电路原理&#xff0c;还能够提升教学效率&#xff0c;培养学生的实践能力。本文将盘点4款最适合教学的电路仿真软件&#xff0c;助您选择最佳的教学利器&#xff0c;同时介绍…

HTTP的三次握手和四次挥手? 我都给你讲清楚 !!!

今天我们来聊聊一个计算机网络中非常基础但又非常重要的概念——HTTP的三次握手和四次挥手。 一、三次握手 首先&#xff0c;我们来聊聊三次握手。三次握手其实是TCP&#xff08;传输控制协议&#xff09;建立连接时的一个过程&#xff0c;而HTTP是基于TCP的&#xff0c;所以我…

centos 环境部署

一、安装redis 1. 升级 GCC 最直接的解决方式是升级你的 GCC 编译器到支持 C11 标准的版本。CentOS 7 默认的 GCC 版本较旧&#xff0c;可能不支持 _Atomic。你可以通过以下步骤升级 GCC&#xff1a; 启用 CentOS 的 Software Collections (SCL) 仓库&#xff0c;该仓库提供了…

学习使用xbox手柄控制小乌龟节点移动

使用xbox手柄控制小乌龟&#xff0c;首先要下载joy功能包&#xff0c;发布sensor_msgs话题也就是手柄和ros通信的话题。 下载的步骤就根据官方文档即可 joy/Tutorials/ConfiguringALinuxJoystick - ROS Wiki 这里我提供一下具体步骤 第一步 安装joy 首先安装对应系统版本的…

Oracle VM(虚拟机)性能监控工具

Oracle VM是一个独立的虚拟化环境&#xff0c;由 Oracle 提供支持和设计&#xff0c;旨在为运行虚拟机提供轻量级、安全的基于服务器的平台。Oracle VM 能够在受支持的虚拟化环境中部署操作系统和应用软件&#xff0c;Oracle VM 将用户和管理员与底层虚拟化技术隔离开来&#x…

Java Web-Tomcat

Web服务器 Web服务器是一个软件程序,对HTTP协议的操作进行封装,使得程序员不必直接对协议进行操作,让Web开发更加便捷。主要功能是“提供网上信息浏览服务”。 Tomcat&#xff0c;是一个 HTTP 服务器。我们只需要在服务器中安装一个Web服务器如Tomcat&#xff0c;然后就可以将…

vue项目使用eletron将打包成桌面应用(.exe)

vue项目使用eletron将打包成桌面应用(.exe) 1.前期准备 两个项目&#xff1a; 1、自己用vue cli创建的项目 2、第二个是去gitee将案例clone下来 案例地址 https://gitee.com/qingplus/electron-quick-start.git 2、测试案例是否可以正常运行 # 进入项目 cd electron-quick-…

考研数学|《1800》《1000》《880》《660》最佳搭配使用方法

直接说结论&#xff1a;基础不好先做1800、强化之前660&#xff0c;强化可选880/1000题。 首先&#xff0c;传统习题册存在的一个问题是题量较大&#xff0c;但难度波动较大。《汤家凤1800》和《张宇1000》题量庞大&#xff0c;但有些题目难度不够平衡&#xff0c;有些过于简单…

TinyEMU源码分析之启动流程

TinyEMU源码分析之启动流程 1 始于0x10002 确定BBL入口点3 mentry.S执行过程4 启动流程小结 本文属于《 TinyEMU模拟器基础系列教程》之一&#xff0c;欢迎查看其它文章。 本文中使用的代码&#xff0c;均为伪代码&#xff0c;删除了部分源码。 1 始于0x1000 我们沿着TinyEMU…

夜晚水闸3D可视化:科技魔法点亮水利新纪元

在宁静的夜晚&#xff0c;当城市的霓虹灯逐渐暗淡&#xff0c;你是否曾想过&#xff0c;那些默默守护着城市安全的水闸&#xff0c;在科技的魔力下&#xff0c;正焕发出别样的光彩&#xff1f;今天&#xff0c;就让我们一起走进夜晚水闸3D模型&#xff0c;感受科技为水利带来的…

AI足球教练上岗利物浦,射门机会提高13%!来自DeepMind,网友:这不公平

梦晨 发自 凹非寺 量子位 | 公众号 QbitAI AI足球教练登上Nature子刊&#xff0c;谷歌DeepMind与利物浦队合作三年打造&#xff1a; 如同AlphaGo颠覆围棋一样&#xff0c;改变了球队制定战术的方式。 像是进攻方把球传给谁更容易创造射门机会&#xff0c;防守方如何调整布阵……

【双指针】Leetcode 四数之和

题目解析 18. 四数之和 这道题的思路和三数之和的思路相同&#xff0c;都是固定一个数&#xff0c;然后在剩下的区间中寻找和为目标值的元组&#xff0c;其次最重要的是要进行去重 算法讲解 1. 完成排序 2. 固定一个数&#xff0c;求剩下区间中三元组的和等于 target - nums[…

用易查分制作承诺书签订,在线手写签名,一键导出打印

假期将至&#xff0c;为积极落实安全管理规定&#xff0c;单位通常需要下发安全承诺书进行签字确认。 易查分可以实现网上下发安全承诺书通知&#xff0c;让查询者进行签名确认&#xff0c;还可以生成PDF&#xff0c;方便打印一人一张的纸质版承诺书&#xff0c;本次就来介绍如…

flask各种版本的项目,终端命令运行方式的实现

目录 写在前面 一、Flask项目的基本结构 二、使用终端命令运行Flask项目 1. 安装Flask 2. 创建Flask应用 3. 配置FLASK_APP环境变量 4. 运行Flask应用 5. 访问Flask应用 三、Flask CLI的其他功能 1. 创建Flask应用 2. 运行开发服务器 3. 清理缓存文件 4. 运行单元…

补题集合2

VJ 409组队赛2 C - Find a Number 没想到这是一个签到题&#xff0c;因为数据量小&#xff0c;状态并不多&#xff0c;所以可以使用 b f s bfs bfs 去跑每一个状态。令 m o d mod mod 是余数&#xff0c; s u m sum sum 是累加和&#xff0c;那么状态最多就只有 500 ∗ 5…

微信小程序的页面交互1

一、page&#xff08;&#xff09;函数 每个页面的s代码全部写入对应的js文件的page&#xff08;&#xff09;函数里面。点击编译&#xff0c;就可以显示js代码的运行效果。注意&#xff0c;每个页面的page&#xff08;&#xff09;函数是唯一的。 page&#xff08;&#xff…

Oracle参数文件详解

1、参数文件的作用 参数文件用于存放实例所需要的初始化参数&#xff0c;因为多数初始化参数都具有默认值&#xff0c;所以参数文件实际存放了非默认的初始化参数。 2、参数文件类型 1&#xff09;服务端参数文件&#xff0c;又称为 spfile 二进制的文件&#xff0c;命名规则…

【CSDN活动】程序员职业生涯的分水岭:年龄还是经验?

&#x1f308;个人主页: 鑫宝Code &#x1f525;热门专栏: 闲话杂谈&#xff5c; 炫酷HTML | JavaScript基础 ​&#x1f4ab;个人格言: "如无必要&#xff0c;勿增实体" 文章目录 程序员职业生涯的分水岭&#xff1a;年龄还是经验&#xff1f;引言技术更新换代…

Vue3中使用paper.js

目录 Paper.js的使用安装Paper引入Paper.js创建画布实例化Paper、Project以及Tool画圆画点和线画矩形导入图片画文字Item组曲线监听键盘事件监听鼠标事件设置动画下载成图片完整代码 Paper.js的使用 安装Paper npm install paper引入Paper.js import * as Paper from "p…

arduino 2.0以上版本上传项目data目录内文件到ESP8266闪存中

开发测试环境&#xff1a; arduino IDE : 2.3.2 开发板 ESP8266 系统&#xff1a;WINDOWS 10 截止目前&#xff0c;arduino版本为2.3.2&#xff0c;在开发项目的时候&#xff0c;发现一个问题&#xff0c;就是项目目录中data内的文件没有办法和主文件.ino一同上传到ESP8266的f…