前言
大家好,我是捡田螺的小男孩。(求个星标置顶)
文章开篇前,先问大家一个问题:delete in子查询,是否会走索引呢?很多伙伴第一感觉就是:会走索引。最近我们有个生产问题,就跟它有关。本文将跟大家一起探讨这个问题,并附上优化方案。
问题复现
MySQL版本是5.7,假设当前有两张表account和old_account,表结构如下:
CREATETABLE`old_account`( `id`int(11)NOTNULLAUTO_INCREMENTCOMMENT'主键Id', `name`varchar(255)DEFAULTNULLCOMMENT'账户名', `balance`int(11)DEFAULTNULLCOMMENT'余额', `create_time`datetimeNOTNULLCOMMENT'创建时间', `update_time`datetimeNOTNULLONUPDATECURRENT_TIMESTAMPCOMMENT'更新时间', PRIMARYKEY(`id`), KEY`idx_name`(`name`)USINGBTREE )ENGINE=InnoDBAUTO_INCREMENT=1570068DEFAULTCHARSET=utf8ROW_FORMAT=REDUNDANTCOMMENT='老的账户表'; CREATETABLE`account`( `id`int(11)NOTNULLAUTO_INCREMENTCOMMENT'主键Id', `name`varchar(255)DEFAULTNULLCOMMENT'账户名', `balance`int(11)DEFAULTNULLCOMMENT'余额', `create_time`datetimeNOTNULLCOMMENT'创建时间', `update_time`datetimeNOTNULLONUPDATECURRENT_TIMESTAMPCOMMENT'更新时间', PRIMARYKEY(`id`), KEY`idx_name`(`name`)USINGBTREE )ENGINE=InnoDBAUTO_INCREMENT=1570068DEFAULTCHARSET=utf8ROW_FORMAT=REDUNDANTCOMMENT='账户表';
执行的SQL如下:
deletefromaccountwherenamein(selectnamefromold_account);
我们explain执行计划走一波,
从explain结果可以发现:先全表扫描 account,然后逐行执行子查询判断条件是否满足;显然,这个执行计划和我们预期不符合,因为并没有走索引。
但是如果把delete换成select,就会走索引。如下:
为什么select in子查询会走索引,delete in子查询却不会走索引呢?
原因分析
select in子查询语句跟delete in子查询语句的不同点到底在哪里呢?
我们执行以下SQL看看
explainselect*fromaccountwherenamein(selectnamefromold_account); showWARNINGS;
show WARNINGS 可以查看优化后,最终执行的sql
结果如下:
select`test2`.`account`.`id`AS`id`,`test2`.`account`.`name`AS`name`,`test2`.`account`.`balance`AS`balance`,`test2`.`account`.`create_time`AS`create_time`,`test2`.`account`.`update_time`AS`update_time`from`test2`.`account` semijoin(`test2`.`old_account`) where(`test2`.`account`.`name`=`test2`.`old_account`.`name`)
可以发现,实际执行的时候,MySQL对select in子查询做了优化,把子查询改成join的方式,所以可以走索引。但是很遗憾,对于delete in子查询,MySQL却没有对它做这个优化。
优化方案
那如何优化这个问题呢?
转载请注明:IT运维空间 » 运维技术 » 生产问题分析!Delete in子查询不走索引?!
发表评论