一个mysql优化技巧的误区

 关于sql优化技巧,大家可能见过N个版本,尤其容易博得初中级程序员的眼球。倘若没有一点分析实践能力,直接将其拿来当作圣经记在心中并实践于工作中,那你极有可能被掉坑。轻则代码运行转圈圈无响应,重则导致项目瘫痪造成经济损失。

        废话不多说,直接上图。


wKioL1jzcDvxLbs2AACDHofFibU963.png


        上面这条技巧粗略看一眼好像也没有什么问题。可事实是这样的吗?

        

        结论当然是否定的。且看实例分析:


        

1
2
3
4
5
6
7
8
9
10
11
12
CREATE  TABLE  `t_auxiliary_info` (
   `id`  int (11) unsigned  NOT  NULL  AUTO_INCREMENT,
   `ac_id` tinyint(3) unsigned  NOT  NULL  COMMENT  '分类ID' ,
   ` name varchar (250)  NOT  NULL  DEFAULT  ''  COMMENT  '名称' ,
   `number`  smallint (6) unsigned  NOT  NULL  DEFAULT  '1'  COMMENT  '编号' ,
   `attr`  varchar (500)  NOT  NULL  DEFAULT  ''  COMMENT  '属性' ,
   `fdbid`  int (10) unsigned  NOT  NULL  COMMENT  '用户ID' ,
   `status` tinyint(1) unsigned  NOT  NULL  DEFAULT  '1'  COMMENT  '状态:1有效,0无效' ,
   `stock_type` tinyint(1) unsigned  NOT  NULL  DEFAULT  '0'  COMMENT  '存货类型:1库存商品,2原材料,3周转材料' ,
   PRIMARY  KEY  (`id`),#请注意这里的索引
   KEY  `uniq_cid_acid` (`fdbid`,`ac_id`)
) ENGINE=InnoDB AUTO_INCREMENT=645101  DEFAULT  CHARSET=utf8 ROW_FORMAT= DYNAMIC


    上面是一张普通的业务表,仔细看表中设置的索引:

1
2
   PRIMARY  KEY  (`id`),#主键索引
   KEY  `uniq_cid_acid` (`fdbid`,`ac_id`)#联合索引


        

        再使用上述的in 或not in 来实践以下,通过explain执行计划工具看看实际效果。(在这里为了公平起见,我不使用主键id,且in操作中的数据不是连续的。)

        

1
2
3
select 
from  t_auxiliary_info 
where  fdbid  in ( '1000' , '1500' , '1234' , '5155' , '6789' , '3423' , '5368' , '245645' );

        

1
在上面的sql中,我们使用包含在联合索引`uniq_cid_acid`中的字段 `fdbid`作为搜索条件

 

       见证奇迹的时刻到了。


wKiom1jzdaPAsg35AAB8Daxo-zs623.png

    

        通过执行计划, 我们可以清晰的看到这条sql的检索类型为简单简单检索,属于范围查询,且已经使用到了索引  uniq_cid_acid,且没有全表扫描(扫描行数为2804,而本表中数据条数为645101)。

        

        由此可以得出结论:不是所有sql中的in查询会全表扫描。这里推翻了in会导致全表扫描的结论。


        那么在什么情况下,使用in操作一样可以使用到索引,不会全表扫描呢?

       答:  in的字段必须是带有索引的字段。

       ps:  in(...) 中的数据最好加上引号,即使字段类型是数字。


        在 看看not in        

1
2
3
select 
from  t_auxiliary_info 
where  fdbid  not  in (1000,1500,1234,5155,6789,3423,5368,245645);

   

        真相在这里:

wKioL1jzeajC3kv_AACBbcpzw3Y564.png


not in确实会全表扫描。










本文转自 hgditren 51CTO博客,原文链接:http://blog.51cto.com/phpme/1916491,如需转载请自行联系原作者
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值