MySQL 索引

在 MySQL 中,索引 是通过特定的数据结构来加速查询操作。MySQL 支持多种类型的索引,其中 B+ 树索引 是最常见的一种。

1. B+ 树索引(B+ Tree Index)

B+ 树的特点:

  1. 所有数据存储在叶子节点

    • 所有的数据都存储在 B+ 树的叶子节点中,非叶子节点仅存储索引。
  2. 叶子节点链表

    • B+ 树的叶子节点通过链表连接,使得在进行范围查询时,可以快速遍历所有匹配的数据。
  3. 多路平衡

    • B+ 树是多路查找树,每个节点可以有多个子节点,保持树的高度较低,从而提高查询效率。
  4. 高度平衡

    • B+ 树保持高度平衡,所有叶子节点都位于树的同一层级,确保查找操作的时间复杂度为 O(log n)。

B+ 树的工作原理:

  • 查找操作:查找时,通过从根节点开始,按照键值的大小比较,逐层向下查找,最终到达叶子节点。在叶子节点中,可以通过链表进行范围查询。
  • 插入和删除操作:插入和删除操作会保持树的平衡,可能会引起节点的分裂或合并,保证 B+ 树始终是平衡的,从而保证查找的效率。

B+ 树和 B 树的区别:

  • 数据存储位置:在 B 树中,数据存储在非叶子节点和叶子节点,而 B+ 树的所有数据都存储在叶子节点。
  • 叶子节点链接:B+ 树的叶子节点通过链表连接,支持范围查询,而 B 树没有这种结构。
  • 查询性能:B+ 树的查询性能通常优于 B 树,尤其是在范围查询时,B+ 树具有更高的效率。

2. 适合加索引的情况

加索引可以显著提高查询效率,特别是在处理大数据量的表和复杂查询时。以下是几种适合加索引的情况:

  1. 查询条件中使用 JOIN 操作

    • 如果两个表通过某个列进行连接查询,这个列应该加上索引。通过索引可以加速连接操作,避免全表扫描。
    SELECT * FROM orders o JOIN customers c ON o.customer_id = c.customer_id;
    

    在这种情况下,orders.customer_idcustomers.customer_id 应该加索引。

  2. 需要进行分组操作(GROUP BY

    • 当查询需要进行分组时(如 COUNT()SUM() 等聚合函数),加上索引可以加速分组操作,尤其是在分组列上加索引。
    SELECT department, COUNT(*) FROM employees GROUP BY department;
    
  3. 涉及到 ORDER BY 的查询

    • 如果查询结果需要排序,在排序的列上加索引可以提高排序效率,避免进行全表扫描和排序。
    SELECT * FROM employees ORDER BY salary DESC;
    
  4. 对于具有唯一性的字段(如用户名、邮箱等)

    • 唯一性字段上加索引可以加速查询,并避免重复值,保证数据一致性。
    CREATE UNIQUE INDEX idx_email ON users (email);
    
  5. 用于防止全表扫描的查询

    • 对于经常在 WHERE 子句中查询的字段,尤其是范围较小的字段,应该考虑加索引来避免全表扫描。
    SELECT * FROM users WHERE age = 25;
    
  6. 使用 DISTINCT 查询

    • 如果查询包含 DISTINCT 操作,索引可以帮助快速排除重复记录,提高查询效率。
    SELECT DISTINCT department FROM employees;
    
  7. 外键列

    • 外键列的索引有助于提高查询性能,并在更新或删除父表记录时加速约束检查。
  8. 经常用作过滤条件的列

    • 如果某列经常作为过滤条件出现,并且该列的数据基数较高(即非唯一性且具有较好的选择性),应该考虑加索引。

3. 什么时候索引会失效?

虽然索引能够提高查询效率,但在某些情况下,索引会失效。以下是几种常见的情况:

  1. 使用 OR 操作符

    • 如果查询条件中使用了 OR 操作符,MySQL 可能会选择其中无法使用索引的条件,导致索引失效。
    SELECT * FROM users WHERE age = 25 OR name = 'Alice';
    
  2. 使用 LIKE 进行模糊查询(尤其是以 % 开头)

    • 对于 LIKE '%text%' 这种查询模式,索引会失效,因为 MySQL 无法通过索引定位模糊匹配的位置。
    SELECT * FROM users WHERE name LIKE '%Alice%';
    
  3. 使用不等号 <>!=

    • 使用不等于操作符通常会导致索引失效,因为 MySQL 不能直接通过索引来确定不等的条件。
    SELECT * FROM users WHERE age <> 30;
    
  4. 数据类型不匹配

    • 当查询条件中的数据类型与列的数据类型不一致时,可能导致索引失效,尤其是在比较字符串与数字时。
    SELECT * FROM users WHERE age = '30';  -- age 是 INT 类型
    
  5. 函数操作

    • 如果在查询条件中使用了函数,MySQL 将无法使用索引,因为函数会影响列的值,使得索引无法有效地参与查询。
    SELECT * FROM users WHERE YEAR(birth_date) = 1990;
    
  6. NULL 值查询

    • 查询含有 NULL 值的字段时,索引可能失效,尤其是在查询条件中使用 IS NULLIS NOT NULL 时。
    SELECT * FROM users WHERE birth_date IS NULL;
    
  7. 部分列的联合索引

    • 在使用多列索引时,如果查询条件没有包含索引的最左边部分,可能导致索引失效。
    CREATE INDEX idx_name_age ON users (name, age);
    SELECT * FROM users WHERE age = 30;  -- 没有涉及 name 列,索引失效
    

总结

  • B+ 树索引:适用于大多数查询,包括精确查找、范围查找和排序操作。它是 MySQL 默认的索引类型,支持高效的查询。
  • 适合加索引的情况
    • 常用的查询条件列,特别是用于 WHEREJOINORDER BYGROUP BY 等的列。
    • 大数据量表、需要排序或唯一性要求的场景。
    • 经常进行连接查询、分组操作、排序和去重操作的列。
  • 索引失效的情况
    • 使用 ORLIKE(以 % 开头)、函数操作、不等于操作等。
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 219,110评论 6 508
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,443评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 165,474评论 0 356
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,881评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,902评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,698评论 1 305
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,418评论 3 419
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,332评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,796评论 1 316
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,968评论 3 337
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,110评论 1 351
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,792评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,455评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,003评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,130评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,348评论 3 373
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,047评论 2 355

推荐阅读更多精彩内容