您当前的位置:首页 > 电脑百科 > 数据库 > MYSQL

为什么MySQL字符串不加引号索引会失效?这个答案是我见过最靠谱

时间:2022-05-09 10:35:29  来源:  作者:洪生鹏

作为一名程序员,在求职面试时,不知你有没有遇到类似这样的问题。

张工是一名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
为什么MySQL字符串不加引号索引会失效?这个答案是我见过最靠谱

 

explain select * from tb_user  where phone = '12345678936'

 

为什么MySQL字符串不加引号索引会失效?这个答案是我见过最靠谱

 

从这两条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内部进行了隐式转换,此次查询会导致全表扫描,所以慢了。

 

总结:

在索引列上进行了函数操作,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;


Tags:MySQL字符串   点击:()  评论:()
声明:本站部分内容及图片来自互联网,转载是出于传递更多信息之目的,内容观点仅代表作者本人,不构成投资建议。投资者据此操作,风险自担。如有任何标注错误或版权侵犯请与我们联系,我们将及时更正、删除。
▌相关推荐
为什么MySQL字符串不加引号索引会失效?这个答案是我见过最靠谱
作为一名程序员,在求职面试时,不知你有没有遇到类似这样的问题。张工是一名java程序员,最近到一家软件公司应聘软件开发岗位,面试官问了他关于MySql索引这样的一个问题。面试官:...【详细内容】
2022-05-09  Search: MySQL字符串  点击:(335)  评论:(0)  加入收藏
mysql字符串截取的秘密武器
在实际的项目开发中有时会有对数据库某字段截取部分的需求,这种场景有时直接通过数据库操作来实现比通过代码实现要更方便快捷些,mysql有很多字符串函数可以用来处理这些需求...【详细内容】
2019-06-14  Search: MySQL字符串  点击:(994)  评论:(0)  加入收藏
▌简易百科推荐
MySQL 核心模块揭秘
server 层会创建一个 SAVEPOINT 对象,用于存放 savepoint 信息。binlog 会把 binlog offset 写入 server 层为它分配的一块 8 字节的内存里。 InnoDB 会维护自己的 savepoint...【详细内容】
2024-04-03  爱可生开源社区    Tags:MySQL   点击:(10)  评论:(0)  加入收藏
MySQL 核心模块揭秘,你看明白了吗?
为了提升分配 undo 段的效率,事务提交过程中,InnoDB 会缓存一些 undo 段。只要同时满足两个条件,insert undo 段或 update undo 段就能被缓存。1. 关于缓存 undo 段为了提升分...【详细内容】
2024-03-27  爱可生开源社区  微信公众号  Tags:MySQL   点击:(18)  评论:(0)  加入收藏
MySQL:BUG导致DDL语句无谓的索引重建
对于5.7.23之前的版本在评估类似DDL操作的时候需要谨慎,可能评估为瞬间操作,但是实际上线的时候跑了很久,这个就容易导致超过维护窗口,甚至更大的故障。一、问题模拟使用5.7.22...【详细内容】
2024-03-26  MySQL学习  微信公众号  Tags:MySQL   点击:(14)  评论:(0)  加入收藏
从 MySQL 到 ByteHouse,抖音精准推荐存储架构重构解读
ByteHouse是一款OLAP引擎,具备查询效率高的特点,在硬件需求上相对较低,且具有良好的水平扩展性,如果数据量进一步增长,可以通过增加服务器数量来提升处理能力。本文将从兴趣圈层...【详细内容】
2024-03-22  字节跳动技术团队    Tags:ByteHouse   点击:(29)  评论:(0)  加入收藏
MySQL自增主键一定是连续的吗?
测试环境:MySQL版本:8.0数据库表:T (主键id,唯一索引c,普通字段d)如果你的业务设计依赖于自增主键的连续性,这个设计假设自增主键是连续的。但实际上,这样的假设是错的,因为自增主键不...【详细内容】
2024-03-10    dbaplus社群  Tags:MySQL   点击:(14)  评论:(0)  加入收藏
准线上事故之MySQL优化器索引选错
1 背景最近组里来了许多新的小伙伴,大家在一起聊聊技术,有小兄弟提到了MySQL的优化器的内部策略,想起了之前在公司出现的一个线上问题,今天借着这个机会,在这里分享下过程和结论...【详细内容】
2024-03-07  转转技术  微信公众号  Tags:MySQL   点击:(33)  评论:(0)  加入收藏
MySQL数据恢复,你会吗?
今天分享一下binlog2sql,它是一款比较常用的数据恢复工具,可以通过它从MySQL binlog解析出你要的SQL,并根据不同选项,可以得到原始SQL、回滚SQL、去除主键的INSERT SQL等。主要...【详细内容】
2024-02-22  数据库干货铺  微信公众号  Tags:MySQL   点击:(54)  评论:(0)  加入收藏
如何在MySQL中实现数据的版本管理和回滚操作?
实现数据的版本管理和回滚操作在MySQL中可以通过以下几种方式实现,包括使用事务、备份恢复、日志和版本控制工具等。下面将详细介绍这些方法。1.使用事务:MySQL支持事务操作,可...【详细内容】
2024-02-20  编程技术汇    Tags:MySQL   点击:(54)  评论:(0)  加入收藏
MySQL数据库如何生成分组排序的序号
经常进行数据分析的小伙伴经常会需要生成序号或进行数据分组排序并生成序号。在MySQL8.0中可以使用窗口函数来实现,可以参考历史文章有了这些函数,统计分析事半功倍进行了解。...【详细内容】
2024-01-30  数据库干货铺  微信公众号  Tags:MySQL   点击:(55)  评论:(0)  加入收藏
mysql索引失效的场景
MySQL中索引失效是指数据库查询时无法有效利用索引,这可能导致查询性能显著下降。以下是一些常见的MySQL索引失效的场景:1.使用非前导列进行查询: 假设有一个复合索引 (A, B)。...【详细内容】
2024-01-15  小王爱编程  今日头条  Tags:mysql索引   点击:(88)  评论:(0)  加入收藏
站内最新
站内热门
站内头条