不清楚MySQL索引为什么会失效,面试官:回去等通知
作为一名程序员,在求职面试时,不知你有没有遇到类似这样的问题。
张工是一名java程序员,最近到一家软件公司应聘软件开发岗位,面试官问了他关于MySql索引这样的一个问题。
面试官:为什么MySQL字符串类型查询时不加引号索引会失效?
张工:MySQL内部进行了隐式转换。
面试官:那为什么会发生隐式转换?
张工:……
对于这个问题张工之前在做项目时也曾遇到,那时候字段明明是加了索引,可不明白为什么还是很慢。后加上引号就正常了,为了赶项目进度,张工也没有再去留意。
现在面试官突然这么一问,张工也说不出个所以然来。
面试官让他回去等通知。
我们知道MySql索引可以加快数据检索速度,这也是使用的索引的最主要原因。但有时候使用不当就会遇到索引失效问题,譬如在MySQL字符串类型查询时不加引号索引会失效,是因为MySQL内部进行了隐式转换。
那为什么会发生隐式转换?又是怎么转换的呢?
今天我们来聊聊关于MySql索引失效的话题。
先来看看一般导致索引失效的有哪些?
1. 最佳左前缀法则
如果一张表的索引有多个,要遵守最佳左前缀法则,即查询从索引的最左前列开始并且不跳过索引中的列。
2.like查询使用不当会导致索引失效
用户表tb_user字段 id,name,age,sex
创建索引为idx_user_name
执行语句:
select * from tb_user where name like ''%xiaoai";
这时候就会导致索引失效
3.在索引列上做加工操作
在索引列上做加工操作,查询时会导致索引失效,从而导致全表扫描。所以,建议不要在索引列上做任何操作。
举个例子,例如订单表tb_order有个索引是dt(日期), 字段数据存放的格式是这样的2021-12-10 这样的,如果有个需求需要根据dt,格式是20220207这样的来查询,这时候就不要对dt进行格式转换了,
select * from tb_order where DATE_FORMAT(dt,'%Y%m%d') ='20220207'
这样索引就失效了。
而是应该对 20220207做格式处理
select * from tb_order where dt=DATE_FORMAT('20220207','%Y-%m-%d')
这样dt索引才不会失效。
4.查询条件两边数据类型不一致
例如我们在订单表tb_order建立了索引idx_order_id,order_id字段类型为varchar
在查询时使用where order_id= 20220207123654100,这样的查询方式会直接造成索引失效。
正确的用法为where order_id=’20220207123654100’
5.范围条件之后的索引会失效
假如有张用户表tb_user,创建的索引为idx_user_name_age_sex_phone 其中name、age、sex都加了索引。
执行语句
select * from tb_user where name = 'xiaoai' and age > 18 and sex =0;
上面这条sql语句只会命中name和age索引,sex索引会失效,复合索引失效需要查看key_len的长度。
再来看一个例子:
explain select * from tb_user where phone = 12345678936
explain select * from tb_user where phone = '12345678936'
从这两条SQL执行的结果我们可以看出,执行第一条SQL没有使用到索引,而执行第二条SQL时使用到了索引。
这是为什么呢?
我们需要先了解下mysql索引优化器工作的原理。选择索引是优化器工作,优化器工作有自己的一套规则,如果等号两边的数据类型不一致,则会发生隐式转换。
基于这条规则,我们回过头看看
explain select * from tb_user where phone = 12345678936;
这条SQL语句执行时就会变为
explain select * from tb_user where cast(phone as signed int) = 12345678936;
由于对索引列进行了函数操作,所以才导致索引失效,从而全表扫描了。
那么问题来了,细心的你不知有没有留意到为什么是把左侧的列转为int类型,而不是把右侧的值转成字符串类型呢?
什么情况下把数字转为字符串,什么情况下把字符串转为数字,优化器它是根据什么规则来进行判断的?其实规则也并不复杂。
若返回1,则把字符串转为数字。
若返回0,则把数字转为字符串。
根据这个规则,我们再回过头看看之前的查询语句
select * from tb_user where phone = 12345678936
select '12345678936' = 12345678936 返回1 所以这时候就把左侧的列值12345678936转成数字。
关于MySql索引失效的问题先简单写到这,建议平时在做项目时还是要多了解下原理,如果你了解其背后的原理,求职面试时和面试官交流起来就会很舒服了,相信能为这次面试加分,提高被录用的概率。
为什么MySQL字符串类型查询时不加引号索引会失效?这是因为要查询的字符串字段没有加引号时,MySQL内部进行了隐式转换,此次查询会导致全表扫描,所以慢了。
总结:
在索引列上进行了函数操作,MySQL内部会进行了隐式转换,导致索引失效,从而产生全表扫描。
由于笔者知识及水平有限,文中错漏之处在所难免,如有不足之处,欢迎交流。
拓展:
索引创建
1、主键索引:
alter table table_name add primary key (column)
2、唯一索引:
alter table table_name add unique (column)
3、普通索引:
alter table table_name add index index_name (column)
4、全文索引:
alter table table_name add fulltext (column)
5、联合索引:
alter table table_name add index index_name (column1,column2,column3)
索引删除
alter table table_name drop index index_name;
-END-
微信公众号:爱开发
不清楚MySQL索引为什么会失效,面试官:回去等通知相关推荐
- 你以为你懂MySQL索引?阿里的面试官:你还太嫩!
相信很多人对于MySQL的索引都不陌生,索引(Index)是帮助MySQL高效获取数据的数据结构. 因为索引是MySQL中比较重点的知识,相信很多人都有一定的了解,尤其是在面试中出现的频率特别高.楼主 ...
- 我们都进入了一个误区,其实面试官 “ 回去等通知 ” 不等于没戏....
大家好,我是二黑,最近出现了下面的一个话题: 怎么判断自己面试是不是凉了? 在群里也有小伙伴在问这些问题,引起了广泛的讨论 下面博主总结了一下知识来聊一聊: 当看到这个话题时,小伙伴们,是不是首先就想 ...
- mysql 索引 美团_美团面试官:说说MySQL的索引
从本文开始,选取牛客网上大厂的面试题,整理出相关内容的知识点. 什么是索引 小学时我们经常用到的字典里有音节索引和部首目录,当我们查字典时,常常用音节索引和部首目录帮助我们提高查找汉字的速度.MySQ ...
- 十几年老Java咳血推荐:MySQL索引原理、失效情况,两万字肝爆,建议收藏!
一.前言 MySQL 作为主流的数据库,是各大厂面试官百问不厌的知识点,但是需要了解到什么程度呢?仅仅停留在 建库.创表.增删查改等基本操作的水平可不够.在面试后端开发的时候,一连几个问题,简直会被问 ...
- 面试官再问你,mysql索引什么时候失效,你偷偷的笑了
索引失效原因总结 复合索引使用左前缀 建立了一个索引分别字段为 a b c where a - and b - order by c,这种使用顺序就符合最佳做前缀,从左向右依次使用了索引 where ...
- MySQL索引原理、失效情况
声明:本文是小编在学习过程中,东拼西凑整理,如有雷同,纯属借鉴. Mysql5.7的版本, InnoDB引擎 目录 1 mysql索引知识 1.1 B+Tree索引 1.2 主键索引和普通索引的区别 ...
- MySQL索引的理解学习,面试不问索引原理就是事务原理
目录 MySQL执行SQL的整体流程 引言, MySQL索引底层学习原因 磁盘介绍(理解磁盘IO) 索引底层数据结构B+树 B+树(聚集索引) B+树(辅助索引) 思考一下为何使用B+树结构, 不是B ...
- MySQL 索引优化及失效场景
MySQL 索引失效场景 左侧模糊查询 模糊查询时(like语句),模糊匹配的占位符位于条件的首部 B+树索引的键值都是排序的,而条件的左侧使用了占位符,会导致无法按照正常的目录进行匹配,从而导致索引 ...
- mysql索引级别快慢_面试小点-MySQL 的两种索引方法如何提高查询速度
MySQL 索引的建立对于 MySQL 的高效运行是很重要的,索引可以大大提高 MySQL 的检索速度. 索引方法 Mysql 的索引方法有两种,BTERR 和 HASH. 散列表(Hash Tabl ...
最新文章
- linux服务器磁盘格式,linux下查看磁盘分区的文件系统格式
- (二)深入浅出TCPIP之再识TCP,理解TCP三次握手(上)
- Linux C目标文件
- LeetCode 1922. 统计好数字的数目(快速幂)
- snmp4j介绍及api使用
- 零成本兼职副业有哪些?
- rgba与16进制颜色格式互转
- x3850用uefi安装Linux7,X3850 X5在uEFI模式下无法安装Centos 6.2的解决办法
- WPF|一个比较简单带点设计的登录界面
- 【181214】VC++动画翻页效果的电话簿程序源代码
- Vue中实现页面截图并上传
- "中国东信杯"广西大学第二届程序设计竞赛(同步赛)
- 星际争霸十大挖掘性操作
- Qt tableview 悬浮提示
- 动漫插画培训班有哪些
- android自定义秒表,Android实现的秒表计时器示例
- 灵魂拷问:Java的可变参数究竟是怎么一回事?
- 小鸟云独享虚拟主机和共享虚拟主机区别对比
- ios 蓝牙开发总结
- FFmpeg转码参数设置