避免写出不走索引的SQL, MySQL

在MySQL中,并不是你建立了索引,并且你在SQL中使用到了该列,MySQL就肯定会使用到那些索引的,有一些情况很可能在你不知不觉中,你就“成功的避开了”MySQL的所有索引。


现假设有t_stu表,age,sname上建立了索引


索引列参与计算


如果where条件中age列中使用了计算,则不会使用该索引


SELECT `sname` FROM `t_stu` WHERE `age`=20;-- 会使用索引
SELECT `sname` FROM `t_stu` WHERE `age`+10=30;-- 不会使用索引!!因为所有索引列参与了计算

SELECT `sname` FROM `t_stu` WHERE `age`=30-10;-- 会使用索引

故,如果需要计算,千万不要计算到索引列,想方设法让其计算到表达式的另一边去。


索引列使用了函数


同样的道理,索引列使用了函数,一样会导致相同的后果


SELECT `sname` FROM `stu` WHERE concat(`sname`,abc) =Jaskeyabc; -- 不会使用索引,因为使用了函数运算,原理与上面相同
SELECT `sname` FROM `stu` WHERE `sname` =concat(Jaskey,abc); -- 会使用索引
索引列使用了Like %XXX


SELECT * FROM `houdunwang` WHERE `uname` LIKE 前缀就走索引% -- 走索引
SELECT * FROM `houdunwang` WHERE `uname` LIKE 后缀不走索引% -- 不走索引
所以当需要搜索email列中.com结尾的字符串而email上希望走索引时候,可以考虑数据库存储一个反向的内容reverse_email


SELECT * FROM `table` WHERE `reverse_email` LIKE REVERSE(%.com); -- 走索引
注:以上如果你使用REVERSE(email) = REVERSE(%.com),一样得不到你想要的结果,因为你在索引列email列上使用了函数,MySQL不会使用该列索引


同样的,索引列上使用正则表达式也不会走索引。


字符串列与数字直接比较


这是一个坑,假设有一张表,里面的a列是一个字符char类型,且a上建立了索引,你用它与数字类型做比较判断的话:


 CREATE TABLE `t1` (`a` char(10));


 SELECT * FROM `t1` WHERE `a`=1 -- 走索引
 SELECT * FROM `t2` WHERE `a`=1 -- 字符串和数字比较,不走索引!
但是如果那个表那个列是一个数字类型,拿来和字符类型的做比较,则不会影响到使用索引


 CREATE TABLE `t2` (`b` int);


 SELECT * FROM `t2` WHERE `b`=1 -- 虽然b是数字类型,和1比较依然走索引
但是,无论如何,这种额外的隐式类型转换都是开销,而且由于有字符和数字比就不走索引的情况,故建议避免一切隐式类型转换


尽量避免 OR 操作


select * from dept where dname=jaskey or loc=bj or deptno=45 --如果条件中有or,即使其中有条件带索引也不会使用。换言之,就是要求使用的所有字段,都必须建立索引
所以除非每个列都建立了索引,否则不建议使用OR,在多列OR中,可以考虑用UNION 替换


select * from dept where dname=jaskey union
select * from dept where loc=bj union
select * from dept where deptno=45
经验分享 程序员 微信小程序 职场和发展