作者:IT王小二
博客:https://itwxe.com
上一篇给小伙伴们讲了关于SQL查询性能优化的相关技巧,一个好的查询SQL离不开合理的索引设计。这篇小二就来唠一唠怎么合理的设计一个索引来优化我们的查询速度,要是有不合理的地方...嗯..

当然啦,开个玩笑,欢迎小伙伴们指正!
索引设计基石是什么呢?
- 小二认为索引设计的基石就是数据表字段类型的合理设计,即选择合适字段类型和设置合适的长度。
- 选择正确的数据类型,那么在字段上建立索引时,一个数据页可以存储更多的索引,一次读取加载到内存的索引个数更多,同时降低B+tree的高度,减少磁盘IO,对提升MySQL的性能提升有着极大的意义。
通常情况下,字段类型的选择是需要根据业务来判断的,通常需要遵循以下几点。
NOT NULL,避免使用NULL。下列各种类型表格内容来自菜鸟教程,权当备忘。
| 类型 | 大小(bytes) | 范围(有符号) | 范围(无符号) | 用途 |
|---|---|---|---|---|
| TINYINT | 1 | (-128, 127) | (0, 255) | 小整数值 |
| SMALLINT | 2 | (-32768, 32767) | (0, 65535) | 大整数值 |
| MEDIUMINT | 3 | (-8388608, 8388 607) | (0, 16777215) | 大整数值 |
| INT或INTEGER | 4 | (-2147483648, 2147483647) | (0, 4294967295) | 大整数值 |
| BIGINT | 8 | (-9233372036854775808, 9223372036854775807) | (0, 18446744073 709551615) | 极大整数值 |
| FLOAT | 4 | (-3.402823466E+38, 1.175494351E-38),0,(1.175494351E-38,3.402823466351E+38) | 0, (1.175494351E-38, 3.402823466E+38) | 单精度浮点数值 |
| DOUBLE | 8 | (1.7976931348623157E+308, 2.2250738585072014E-308), 0, (2.2250738585072014E-308, 1.7976931348623157E+308) | 0, (2.2250738585072014E-308, 1.7976931348623157E+308) | 双精度浮点数值 |
| DECIMAL | 对DECIMAL(M,D) ,如果M>D,为M+2否则为D+2 | 依赖于M和D的值 | 依赖于M和D的值 | 小数值 |
优化建议:
注意: INT(2)设置的为显示宽度,而不是整数的长度,需要配合ZEROFILL使用 。
CREATE TABLE user( id TINYINT(2) UNSIGNED);例如id设置为 TINYINT(2) UNSIGNED,表示无符号,可以存储的最大数值为255,其中TINYINT(2)没有配合ZEROFILL实际没有任何意义,例如插入数字200,长度虽然超过了两位,但是这个时候是可以插入成功的,查询结果同样为200;插入数字5时,同样查询结果为5。
CREATE TABLE user( id TINYINT(2) UNSIGNED ZEROFILL);而TINYINT(2)配合ZEROFILL后,当插入数字5时,实际存储的还是5,不过在查询是MySQL会在前面补上一个0,即查询出来的实际为05。
| 类型 | 大小(bytes) | 用途 |
|---|---|---|
| CHAR | 0-255 | 定长字符串,char(n)当插入的字符数不足n时(n代表字符数),插入空格进行补充保存。在进行检索时,尾部的空格会被去掉。 |
| VARCHAR | 0-65535 | 变长字符串,varchar(n)中的n代表最大字符数,插入的字符数不足n时不会补充空格 |
| TINYBLOB | 0-255 | 不超过 255 个字符的二进制字符串 |
| TINYTEXT | 0-255 | 短文本字符串 |
| BLOB | 0-65535 | 二进制形式的长文本数据 |
| TEXT | 0-65535 | 长文本数据 |
| MEDIUMBLOB | 0-16777215 | 二进制形式的中等长度文本数据 |
| MEDIUMTEXT | 0-16777215 | 中等长度文本数据 |
| LONGBLOB | 0-4294967295 | 二进制形式的极大文本数据 |
| LONGTEXT | 0-4294967295 | 极大文本数据 |
优化建议:
idx_name_age_position(name(20), age, position),取前20个字符作为索引,但是这种情况下因为是不完全字段,所以order by name asc或者group by name 排序过程无法使用索引排序。当然需要保证数据的精确性和查找速度,最优的方案就是使用全文搜索引擎ES了。| 类型 | 大小(bytes) | 范围 | 格式 | 用途 |
|---|---|---|---|---|
| DATE | 3 | 1000-01-01 到 9999-12-31 | YYYY-MM-DD | 日期值 |
| TIME | 3 | '-838:59:59' 到 '838:59:59' | HH:MM:SS | 时间值或持续时间 |
| YEAR | 1 | 1901 到 2155 | YYYY | 年份值 |
| DATETIME | 8 | 1000-01-01 00:00:00 到 9999-12-31 23:59:59 | YYYY-MM-DD HH:MM:SS | 混合日期和时间值 |
| TIMESTAMP | 4 | 1970-01-01 00:00:00 到 2038-01-19 03:14:07 (格林尼时间) | YYYYMMDD HHMMSS | 混合日期和时间值,时间戳 |
优化建议:
YYYY-MM-DD。CURRENT_TIMESTAMP作为(MySQL5.6以后),MySQL会自动返回记录插入的当前确切时间。不过需要注意的是校准MySQL运行环境的时间和时区,比如Linux时间或者docker容器的时间和时区。通常来说,考虑好表中每个字段应该使用什么类型和长度,建完表需要做的事情不是马上建立索引,而是先把相关主体业务开发完毕,然后把涉及该表的SQL都拿出来分析之后再建立索引。
尽量少建立单值索引(唯一索引除外),应当设计一个或者两三个联合索引,让每一个联合索引都尽量去包含SQL语句中的where、order by、group by的字段,同时确保联合索引的字段顺序尽量满足SQL查询的最左前缀原则。
索引基数是指这个字段在表里总共有多少个不同的值,比如一张表总共100万行记录,其中有个性别字段,性别一共有三个值:男、女、保密,那么该字段的基数就是3。
如果对这种小基数字段建立索引的话,因为索引树中只有男、女、保密三个值,根本没法进行快速的二分查找,同时还需要回表查询,还不如全表扫描嘞。
一般建立索引,尽量使用那些基数比较大的字段,那么才能发挥出B+树快速二分查找的优势来。
在where和order by出现索引设计冲突时,是优先针对where去设计索引?还是优先针对order by设计索引?
通常情况下都是优先针对where来设计索引,因为通常情况下都是先where条件使用索引快速筛选出来符合条件的数据,然后对进行筛选出来的数据进行排序和分组,而where条件快速筛选出来的的数据往往不会很多。
对生产实际运行过程中,或者测试环境大数据量测试过程中发现的慢查询SQL进行特定的索引优化、代码优化等策略。
终于轮到实战了,小二最喜欢实战了。
写到这里不得不吐槽一下,这个金三银四的跳槽季节,年前提离职了,结果离职还没办完就封村整整两个礼拜了,呜呜呜...
上节小二就提到会有个很有意思的小案例,那么在疫情当下,门都出不去的日子,感觉这个例子更有意思了,咱们来讨论一下各种社交平台怎么做的用户信息搜索呢。
社交平台有一个小伙伴们都喜欢的功能,搜索好友信息,比如小二熟练的点开省份...城市..性别..年龄..身高...
咳咳咳...小二怎么可能干这种事情,小二的心里只有代码,嗯...没错,就是这样。

这个就可以说是对于用户信息的查询筛选了,通常这种表都是非常大数据量的,在不考虑分库分表的情况下,怎么通过索引配合SQL来优化呢?
通常我们在编写SQL是会写出类似如下的SQL来执行,有where、order by、limit等条件来查询。
select xx from user where xx=xx and xx=xx order by xx asc limit xx,xx;那么接下来小二一个一个慢慢增加字段来分析分析,怎么根据业务场景来设计索引。
例如通常小伙伴们都会优先筛选出自己所属城市和性别的人,那么该怎么设计索引呢?
where province = xx and city = xx and sex = xx
针对这种情况,很简单,设计一个联合索引(provice, city, sex)就完事了。
那么这个时候小伙伴肯定又要瞅瞅年龄段了,嘿嘿??
where province = xx and city = xx and sex = xx and age >= 18 and age <= 28
那么这时候有小伙伴就会说了,很简单啊,范围字段放最后咱还是知道的,联合索引改成(provice, city, sex, age)不就可以了。
嗯,是的,这么干没毛病,但是小伙伴们有没有想过有些人万一既喜欢帅哥又喜欢美女,别想歪了哈...,挺多小姐姐就既喜欢帅哥又喜欢美女的。
那么这个时候小姐姐就不搜索性别了,那么这个时候联合索引只能用到前两个字段了,那么不符合咱们的专业标准啊,咋办呢?这时候还是有办法的,咱们只需要动动小脑袋改改SQL就行了,在没有选择性别时判断一下,改成下面这样就可以了。
province=xx and city=xx and sex in ('male','female') and age >= 18 and age <= 28那么有爱好之类的其他等值字段。
province=xx and city=xx and sex in ('male','female') and hobby = 1 and xx = xx and age >= 18 and age <= 28
咋办嘞,同样往联合索引里面塞,例如(provice, city, sex, hobby, xx, age)。
那么如果还有范围查询,比如身高、体重范围和最后登录时间等等。
针对这种多个范围查询的话,为了比较好的利用索引,在业务允许的情况下可以使用固定范围,然后数据库字段存储范围标识就可以了,这样就转化为了等值匹配,就可以很好地利用索引了。
例如最后登录时间字段不记录最后登录时间,而是记录设置字段 is_login_within_seven_days在7天内有登录则为1,否则为0,最后索引设计成(provice, city, sex, hobby, xx, is_login_within_seven_days, age)。
那么根据场景最后设计出来的这个索引可能已经可以覆盖大部分的查询流量了,那么如果还有其他一部分热度比较高的查询怎么办呢,办法也很简单啊,再加一两个索引即可。
例如通常会查询这个城市比较受欢迎(评分:score)的小姐姐,这时候添加一个联合索引(provice, city, sex, score)那么就可以了。
可以看出,索引时必须结合场景来设计的,思路就是尽量用不超过3个复杂的联合索引来抗住大部分的80%以上的常用查询流量,然后再用一两个二级索引来抗下一些非常用查询流量。
以上就是小二要给大家分享的索引设计,如果能动动你发财的小手给小二点个免费的赞就更好啦~
下篇小二就来讲讲MySQL事务和锁机制。