mysql中delete in子查询不走索引问题怎么解决

网友投稿 467 2023-11-22

mysql中delete in子查询不走索引问题怎么解决

本文小编为大家详细介绍“mysql中delete in子查询不走索引问题怎么解决”,内容详细,步骤清晰,细节处理妥当,希望这篇“mysql中delete in子查询不走索引问题怎么解决”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。

问题复现

mysql中delete in子查询不走索引问题怎么解决

MySQL版本是5.7,假设当前有两张表account和old_account,表结构如下:

CREATE TABLE `old_account` (   `id` int(11NOT NULL AUTO_INCREMENT COMMENT 主键Id,   `name` varchar(255DEFAULT NULL COMMENT 账户名,   `balance` int(11DEFAULT NULL COMMENT 余额,   `create_time` datetime NOT NULL COMMENT 创建时间,   `update_time` datetime NOT NULL ON UPDATE CURRENT_TIMESTAMP COMMENT 更新时间,   PRIMARY KEY (`id`),   KEY `idx_name`(`name`USING BTREE ) ENGINE=InnoDB AUTO_INCREMENT=1570068 DEFAULT CHARSET=utf8 ROW_FORMAT=REDUNDANTCOMMENT=老的账户表; CREATE TABLE `account` (   `id` int(11NOT NULL AUTO_INCREMENT COMMENT 主键Id,   `name` varchar(255DEFAULT NULL COMMENT 账户名,   `balance` int(11DEFAULT NULL COMMENT 余额,   `create_time` datetime NOT NULL COMMENT 创建时间,   `update_time` datetime NOT NULL ON UPDATE CURRENT_TIMESTAMP COMMENT 更新时间,   PRIMARY KEY (`id`),   KEY `idx_name` (`name`USING BTREE ) ENGINE=InnoDB AUTO_INCREMENT=1570068 DEFAULT CHARSET=utf8 ROW_FORMAT=REDUNDANTCOMMENT=账户表;

执行的SQL如下:

delete from account where name in (select name from old_account);

我们explain执行计划走一波,

从explain结果可以发现:先全表扫描 account,然后逐行执行子查询判断条件是否满足;显然,这个执行计划和我们预期不符合,因为并没有走索引

但是如果把delete换成select,就会走索引。如下:

为什么select in子查询会走索引,delete in子查询却不会走索引呢?

原因分析

select in子查询语句跟delete in子查询语句的不同点到底在哪里呢?

我们执行以下SQL看看

explain select * from account where name in (select name from old_account); show WARNINGS;

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`

semi join (`test2`.`old_account`)

where (`test2`.`account`.`name` = `test2`.`old_account`.`name`)

可以发现,实际执行的时候,MySQL对select in子查询做了优化,把子查询改成join的方式,所以可以走索引。但是很遗憾,对于delete in子查询,MySQL却没有对它做这个优化。

优化方案

那如何优化这个问题呢?通过上面的分析,显然可以把delete in子查询改为join的方式。我们改为join的方式后,再explain看下:

可以发现,改用join的方式是可以走索引的,完美解决了这个问题。

实际上,对于update或者delete子查询的语句,MySQL官网也是推荐join的方式优化

其实呢,给表加别名,也可以解决这个问题哦,如下:

explain delete a from account as a where a.name in (select name from old_account)

为什么加个别名就可以走索引了呢?

what?为啥加个别名,delete in子查询又行了,又走索引了?

我们回过头来看看explain的执行计划,可以发现Extra那一栏,有个LooseScan

LooseScan是什么呢? 其实它是一种策略,是semi join子查询的一种执行策略。

因为子查询改为join,是可以让delete in子查询走索引;加别名呢,会走LooseScan策略,而LooseScan策略,本质上就是semi join子查询的一种执行策略。

因此,加别名就可以让delete in子查询走索引啦!

读到这里,这篇“mysql中delete in子查询不走索引问题怎么解决”文章已经介绍完毕,想要掌握这篇文章的知识点还需要大家自己动手实践使用过才能领会,如果想了解更多相关内容的文章,欢迎关注行业资讯频道。

版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。

上一篇:Redis实现限流器的方法有哪些
下一篇:mdf文件如何导入数据库
相关文章

 发表评论

暂时没有评论,来抢沙发吧~