隐式字符集转换导致索引失效问题怎么避免和解决
Admin 2022-07-28 群英技术资讯 305 次浏览
在索引优化时,经常会看到的一句话:如果索引字段出现隐式字符集转换的话,那么索引将失效,进而转为全表扫描,查询效率将大大降低,要避免出现隐式字符集转换;
在此我想问问同学们:大家知道为什么隐式字符集转换会导致索引失效吗?
实际场景中有没有遇到过隐式字符集转换导致索引失效的场景,具体排查的过程;
本文主线:由上面的两个问题牵引出了本文的主线;
简单描述下隐式字符集转换导致索引失效的原因
然后模拟实际场景排查隐式字符集转换导致索引失效的过程
隐式字符集转换导致索引失效的原因MySQL索引的数据结构是 B+Tree,想要走索引查询必须要满足其 最左前缀原则 ,否则无法通过索引树进行查找,只能进行全表扫描;
例如:下面的这个SQL由于在 索引字段 上使用函数进行运算,导致索引失效
se le ct *上面的这个SQL怎么改造才能使索引生效呢?如下所示:
se le ct * f r o m t_user W H E RE name like '李彤%'通过上面的小例子可以知道,如果在索引字段上使用函数运算,则会导致索引失效,而索引字段的 隐式字符集转换 由于MySQL会自动的在索引字段上加上 转换函数 ,进而会导致索引失效;
那接下来我们就通过模拟的实际场景来具体看看是不是由于MySQL自动给加上了转换函数而导致索引失效的;
模拟场景 + 问题排查由于导致索引失效的原因有很多,如果自己写的SQL怎么看都没问题,但是通过查看执行计划发现就是没有走索引查询,此时就会让很多人陷入困境,这到底是怎么导致的呢?
此时本文重点将要讲述的工具就要闪亮登场啦: expl ain EX TE ND ED + show warnings ;
使用这个工具可以将执行的SQL语句的一些扩展信息展示出来,这些扩展信息就包括:MySQL优化时可能会添加上字符集转换函数,使得字符集不匹配的SQL可以正确执行下去;
下面就来具体聊聊 expl ain EX TE ND ED + show warnings 的使用;
模拟隐式字符集转换的场景:首先创建两个字符集不一样的表:
然后使用存储过程构造数据:
注意:在构造数据时,记得将 t_employees 表中的 de_no 字段值构造的 离散些 ,因为如果索引字段值的 区分度很低 的话,那么MyQSL优化器通过采样统计分析时,发现索引查询和全表扫描性能差不多,就会直接进行全表扫描了;
索引失效的查询SQL语句:
将表和数据构造完后,我们使用SQL语句进行查询下,然后再看看其执行计划;
expl ain se le ct * f r o m t_department a LE FT JO IN t_employees b on a.de_no = b.de_no W H E RE a.id = 16其执行计划如下:
发现 t_employees 表中的 de_no 字段有索引,但是没有走索引查询,type=ALL 走的全表扫描,但是通过查看SQL语句发现其没有问题呀,表面看上去都是满足走索引查询的条件呀,排查到这发现遇到了困境,苦恼啊!
还好,通过在网络世界上遨游,最终发现了 expl ain EX TE ND ED + show warnings 利器,利用它快速发现了索引失效的根本原因,然后快速找到了解决方案;
下面就来聊聊这个利器的具体使用,开森!
使用利器快速排查问题:
注意:expl ain 后面跟的关键字 EX TE ND ED(扩展信息) 在MySQL5.7及之后的版本中废弃了,但是该语法仍被识别为向后兼容,所以在5.7版本及后续版本中,可以不用在 expl ain 后面添加 EX TE ND ED 了;
具体使用方法如下:
①、首先在MySQL的可视化工具中打开一个 命令列介面 :工具 --> 命令列介面
②、然后输入下面的SQL并按回车:
expl ain EX TE ND ED se le ct * f r o m t_department a LE FT JO IN t_employees b on a.de_no = b.de_no W H E RE a.id = 4019;③、然后紧接着输入命令 show warnings; 并回车,会出现如下图所示内容:
通过展示出的执行SQL扩展信息,发现MySQL在字符集不一致时自动添加上字符集转换函数,因为是在 索引字段 de_no 上添加的转换函数,所以就导致了索引失效;
而如果我们没看扩展信息的话,那么可能直到我们查看表结构的时候才会发现是由于字符集不一致导致的,这样就会花费很多的时间;
扩展:隐式类型转换咱们聊完上面的隐式字符集转换导致索引失效的情况,再来简单聊聊另一种 隐式类型转换 导致索引失效的情况;
隐式类型转换:简单的说就是字段的类型与其赋值的类型不一致时会进行隐式的转换;
小例如下:
se le ct * f r o m t_employees W H E RE em_name = 123;上面的SQL中 em_name 为索引字段,字段类型是 varchar,为其赋 int 类型的值时,会发现索引失效,这里也可以通过 expl ain EX TE ND ED + show warnings 查看
免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:mmqy2019@163.com进行举报,并提供相关证据,查实之后,将立刻删除涉嫌侵权内容。
猜你喜欢
这篇文章主要介绍SQL图书管理系统数据库的设计,会通过数据分析,实体图,E-R图,数据表关系图以及SQL代码的步骤来介绍数据库的设计,对大家学习有一定的帮助,感兴趣的朋友就继续往下看吧。
SQL Server和MySQL中的Date函数的用法是什么?注意:当我们处理日期时,最困难的任务可能是确保插入日期的格式与数据库中日期列中的格式相匹配。只要您的数据仅包含日期的一部分,运行查询就不会成为问题。然而,当涉及到时间时,情况会稍微复杂一些。
这篇文章主要介绍了SQL Server实现将特定字符串拆分并进行插入操作的方法,涉及SQL Server的循环、遍历、判定及插入等相关操作技巧,需要的朋友可以参考下
本文列表中是 SQL 中所有与日期和时间相关的重要函数。你所用的 RDBMS 可能会支持更多其他的函数。下文的讲解详细,步骤过程清晰,对大家进一步学习和理解相关知识有一定的帮助。有这方面学习需要的朋友就继续往下看吧!
这篇文章主要介绍了SQL Server表中添加新列并添加描述的相关资料,非常不错,具有参考借鉴价值,需要的朋友可以参考下
成为群英会员,开启智能安全云计算之旅
立即注册Copyright © QY Network Company Ltd. All Rights Reserved. 2003-2020 群英 版权所有
增值电信经营许可证 : B1.B2-20140078 粤ICP备09006778号 域名注册商资质 粤 D3.1-20240008