正确的顺序依赖于使用该索引的查询,并且同时需要考虑如何更好的满足排序和分组的需要(本节适用于B-Tree索引)
在一个多列B-Tree索引中,索引列的顺序意味着首先按照最左列进行排序,其次是第二列,等等.所以,索引可以按照升序或者降序进行扫描,以满足精确符合列顺序的ORDER BY,GROUP BY和DISTINCT等子句的查询需要
所以多列索引的列顺序很重要.在三星索引系统中,列顺序也决定了一个索引是否能够成为一个真正的三星索引
对于如何选择索引的列顺序有一个经验法则,将选择性最高的列放到索引最前列.这个建议在大部分场景下有用,但通常不如避免随机IO和排序那么重要,考虑问题需要全面
当不需要考虑排序和分组时,将选择性最高的列放在前面通常是好的.然而,性能不只是依赖于所有索引列的选择性(整体基数),也和查询条件的具体值有关,也就是和值的分布有关
以下面的查询为例子
~~~
select * from payment where staff_id = 2 and customer_id = 584;
~~~
是应该创建一个(staff_id,customer_id)索引还是应该颠倒下顺序?可以跑一些查询来确定在这个表中值的分布情况,并确定那个列的选择性更高.先用下面的查询预测下,看看where条件的分支对应的数据基数有多大
~~~
select sum(staff_id = 2), sum(customer_id=584) from payment;
~~~
根据结果,我们来决定把什么放到前面
这样做有一个地方需要注意,查询的结果非常依赖于选定的具体值.如果按照上述优化,可能对其他一些条件值的查询不公平,服务器的整体性能可能变得更糟,或者其他查询运行变得不如预期
如果是从比如pt-query-digest这样的工具的报告中提取最差查询,那么再按上述办法选定的索引顺序往往是非常高效的
![](https://box.kancloud.cn/22f3a91db8177e6ccdeacaa0c77410b0_564x240.png)
最后,尽管关于选择性和基数的经验法值得研究,但是别忘了where子句中的排序,分组和范围条件等其他因素,这些因素可能对查询的性能造成非常大的影响
- 书列表
- laravel框架关键技术
- 第一章 组件化开发与composer使用
- 简介
- composer
- 添加路由组件
- 添加控制器模块
- 添加模型组件
- 添加视图组件
- 第三章 laravel框架中常用的php语法
- 匿名函数
- 文件包含
- 魔术方法
- 魔术常量
- 反射
- 后期静态绑定
- traits
- 第四章 laravel框架中使用的HTTP协议基础
- HTTP协议
- 数据库
- 数据迁移
- 第六章 laravel框架中的设计模式
- IOC模式
- php核心技术与最佳实践
- 第一章面向对象核心
- 反射
- 简单ORM
- 异常和错误
- 接口
- 第二章,面向对象设计
- 设计原则
- 单一职责
- 接口隔离
- 开放封闭
- 替换原则
- 依赖倒置
- linux是怎么写的呢?
- 第三章 正则表达
- 认识正则
- 第四章 php网络技术应用
- HTTP协议详解
- php和http相关函数
- 垃圾信息防御措施
- 现代操作系统
- 引论
- sql必知必会
- 限制结果
- 按位置排序
- where求职顺序
- IN操作符
- like
- 函数
- group by
- 组合查询
- 插入检索出的数据
- 视图
- 高性能mysql
- 第一章节 mysql架构与历史
- mysql架构逻辑图
- 连接与管理
- 优化与运行
- 读写锁
- 锁粒度
- 表锁(table lock)
- 行级锁(row lock)
- ACID
- 隔离级别
- 死锁
- 隐式和显式锁定
- 多版本并发控制
- Innodb概览
- 第四章节 Schema与数据类型优化
- 选择优化的数据类型
- 日期和时间类型
- 标识列
- 特殊类型数据
- 表设计中的缺陷
- 范式
- 计数器表
- 第五章 创建高性能索引
- 索引基础
- 索引类型
- 索引的优点
- 高性能索引策略
- 选择合适的索引列顺序
- 聚簇索引
- 顺序的主键什么时候会造成更坏的后果
- 覆盖索引
- 使用索引扫描来做排序
- 压缩索引
- 冗余和重复索引
- 索引和锁
- 支持多种过滤条件
- 什么是范围条件
- 优化排序
- 维护索引和表
- 表损坏
- 减少索引和数据的碎片
- 第六章 查询性能优化
- 扫描的行数和访问类型
- 重构查询方式
- 查询执行的基础
- 重构-改善既有代码设计
- 第一章-重构
- 什么是重构
- 第一个案列
- 重构第一步
- 王垠博客
- 多态取代价格相关逻辑