在线精品99_中国九九盗摄偷拍偷看_91免费版在线观看_91.app_91高清视频在线_99热最新网站

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

127次阅读
没有评论

共计 2483 个字符,预计需要花费 7 分钟才能阅读完成。

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

问题复现

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

CREATE TABLE `old_account` ( `id` int(11) NOT NULL AUTO_INCREMENT COMMENT  主键 Id ,
 `name` varchar(255) DEFAULT NULL COMMENT  账户名 ,
 `balance` int(11) DEFAULT 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=REDUNDANT COMMENT= 老的账户表 
CREATE TABLE `account` ( `id` int(11) NOT NULL AUTO_INCREMENT COMMENT  主键 Id ,
 `name` varchar(255) DEFAULT NULL COMMENT  账户名 ,
 `balance` int(11) DEFAULT 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=REDUNDANT COMMENT= 账户表 

执行的 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 子查询不走索引问题怎么解决”文章已经介绍完毕,想要掌握这篇文章的知识点还需要大家自己动手实践使用过才能领会,如果想了解更多相关内容的文章,欢迎关注丸趣 TV 行业资讯频道。

正文完
 
丸趣
版权声明:本站原创文章,由 丸趣 2023-07-13发表,共计2483字。
转载说明:除特殊说明外本站除技术相关以外文章皆由网络搜集发布,转载请注明出处。
评论(没有评论)
主站蜘蛛池模板: 免费在线观看黄视频 | 日本十八禁视频无遮挡 | 精品一区二区在线观看 | 日本久久免费大片 | heyzo国产亚洲高清 | h免费看| 久久r视频 | 自拍偷自拍亚洲精品被多人伦好爽 | 国内久久久久久久久久 | 欧美一区二区三区在线 | 日韩一区二区视频在线观看 | 亚洲最大情网站在线观看 | 黑鬼大战白妞高潮喷白浆 | 久久99视频 | 日木av无码专区亚洲av毛片 | 欧美精品国产一区二区 | jizz 日本亚洲 | 国产伦精品一区二区三区 | 无码人妻精品一区二区三区在线 | 欧美va放荡人妇大片 | 福利一二三区 | 无限资源欧美 | 免费一级a毛片在线播放视 免费一级e一片在线播放 | 久久男人av资源网站无码软件 | 蜜臀国产在线视频 | 私人情侣网站中文 | 欧美成人tv在线观看免费 | www.精品 | 欧美日韩亚洲m码色帝国 | 久久99亚洲网美利坚合众国 | 很黄很色裸乳视频网站 | 亚洲国产精品久久久久久无码 | 高清一级毛片一本到免费观看 | 国内亚州视频在线观看 | 国产做爰视频免费看 | 中国丰满少妇人妻xxx性董鑫洁 | 韩国精品欧美一区二区三区 | 91白丝制服被啪到喷水在线 | 久久乐国产精品亚洲综合m3u8 | 日本理论片午午伦夜理片2021 | 欧美精品在线免费 |