mysql查询索引的过程

综述

首先需要理解以下概念:

  1. B+Tree、聚簇索引、二级索引、稀疏索引
  2. mysql page的结构

其次,总体而言可以将获取数据的类型分为:

  1. 命中了索引,可以直接从聚簇索引上面获取数据,或者通过二级索引定位到聚簇索引,接着获取数据;
  2. 完全没有命中索引,mysql需要扫描所有数据页(也就是聚簇索引B+树的叶子节点);

具体而言,就是分析where条件的具体写法,也就是常说的索引失效的情况:

  1. 查询过程不符合B+树索引结构:

    1. 条件中有or;
    2. 使用!=,<, > 等范围查询;
    3. 使用is null和is not null;
    4. 全文匹配的like,只有%写在最后面才可以走索引。
  2. 在索引列上搞骚操作:

    1. 在索引列上进行计算;
    2. 在索引列上进行隐式转换;
    3. 在索引中使用函数;

不确定的情况下可以使用explain语句,分析索引使用情况,主要看type、keys和extra字段,keys不做赘述。(分析性能可以看rows,即扫描了多少行)

  • type字段:

    1. 没有join操作,type可以是all、index、range、const
    2. 有join操作,type可以是eq_ref和ref
  • extra字段:当type是index的时候,说明用到了索引,extra可以有3个值:

    1. using where:等同于type=all,where是在取完所有数据之后才过滤;
    2. using index:可以直接在索引树上完成检索,无需访问实际的行数据。

mysql的数据也并非所有情况下都在叶子节点,当数据类型是blob或text且超过page size的一半(通常是8k,默认的page size是16k,这个值可以配置为4k,8k,16k,32k,64k),blob字段的值会被放到其他页,索引页只留下blob的前768个字节。

所谓的回表

默认情况下,mysql会用id作为聚簇索引;在没有id或者id不唯一的情况下,才会使用unique key作为索引。
假设有二级索引且有id作为聚簇索引,mysql会创建两颗B+树,查询二级索引会先去二级索引的B+树的叶子节点找到id,再去聚簇索引查到具体的数据页,就是俗称的回表。
图片

关于join的常规优化小表join大表

原理:join在mysql的实现可以理解为一个双循环,小表join大表减少外层循环的次数,从而减少总的笛卡尔积次数。
通常情况下left join会用左表作为驱动表,所以左表用小表;right join会用右表做驱动表;
例外情况:如果两表之中有一个能用到索引,会用能用到索引那张表做驱动表。

1 声望
0 粉丝
0 条评论
推荐阅读
腾讯云长沙一面&中金二面复盘
ThreadPool的底层原理。假设有一个coreSize=2m, maxSize=4, queue.size()=6的线程,当线程数到达6的时候,继续添加线程会发生什么?

izzzyx阅读 204

万字长文~vue+express+mysql带你彻底搞懂项目中的权限控制(附所有源码)
所谓的权限,其实指的就是:用户是否能看到,以及是否允许其对数据进行增删改查的操作,因为现在开发项目的主流方式是前后端分离,所以整个项目的权限是后端权限控制搭配前端权限控制共同实现的

水冗水孚11阅读 1.5k

花了几个月时间把 MySQL 重新巩固了一遍,梳理了一篇几万字 “超硬核” 的保姆式学习教程!(持续更新中~)
MySQL 是最流行的关系型数据库管理系统,在 WEB 应用方面 MySQL 是最好的 RDBMS(Relational Database Management System:关系数据库管理系统)应用软件之一。

民工哥11阅读 1.1k

封面图
一次偶然机会发现的MySQL“负优化”
今天要讲的这件事和上述的两个sql有关,是数年前遇到的一个关于MySQL查询性能的问题。主要是最近刷到了一些关于MySQL查询性能的文章,大部分文章中讲到的都只是一些常见的索引失效场合,于是我回想起了当初被那个...

骑牛上青山8阅读 2.3k评论 2

初学后端,如何做好表结构设计?
这篇文章介绍了设计数据库表结构应该考虑的4个方面,还有优雅设计的6个原则,举了一个例子分享了我的设计思路,为了提高性能我们也要从多方面考虑缓存问题。

王中阳Go3阅读 789评论 2

封面图
2023最新MySQL高频面试题汇总
本文已经收录到Github仓库,该仓库包含计算机基础、Java基础、多线程、JVM、数据库、Redis、Spring、Mybatis、SpringMVC、SpringBoot、分布式、微服务、设计模式、架构、校招社招分享等核心知识点,欢迎star~

程序员大彬3阅读 1k

Mysql索引覆盖
通常情况下,我们创建索引的时候只关注where条件,不过这只是索引优化的一个方向。优秀的索引设计应该纵观整个查询,而不仅仅是where条件部分,还应该关注查询所包含的列。索引确实是一种高效的查找数据方式,但...

京东云开发者2阅读 969

封面图
1 声望
0 粉丝
宣传栏