记踩到 MySQL in 子查询的“坑”

前言

mysql是项目中常用的数据库,其中in查询也是很常用。最近项目调试过程中,遇到一个出乎意料的select查询,竟然用了33秒!

一、表结构

1. userinfo 表

记踩到 MySQL in 子查询的“坑”

2. article 表

记踩到 MySQL in 子查询的“坑”

select*fromuserinfowhereidin(selectauthor_idfromartilcewheretype=1);

登录后复制

大家第一眼看到上面的SQL时,可能都会觉得这是一个很简单的子查询。先把author_id查出来,再用in查询一下。

如果有相关索引会非常快的,拆解来讲就是以下这样的:

1.selectauthor_idfromartilcewheretype=1;  2.select*fromuserinfowhereidin(1,2,3);

登录后复制

但是事实是这样的:

mysql> select count(*) from userinfo;

登录后复制

记踩到 MySQL in 子查询的“坑”

mysql> select count(*) from article;

登录后复制

记踩到 MySQL in 子查询的“坑”

mysql> select id,username from userinfo where id in (select author_id from article where type = 1);

登录后复制

记踩到 MySQL in 子查询的“坑”

33 秒为什么会这么慢呢?

三、问题原因

官方文档解释:in 子句在查询的时候有时会被转换为 exists 的方式来执行,变成逐条记录进行遍历(版本 5.5 中存在,5.6 中已做优化)。

记踩到 MySQL in 子查询的“坑”

参考:

https://dev.mysql.com/doc/refman/5.5/en/subquery-optimization.html

四、解决方式(版本5.5)

1. 使用临时表

select id,username from userinfowhere id in (select author_id from(select author_id from article where type = 1) as tb);

登录后复制

记踩到 MySQL in 子查询的“坑”

2. 使用 join

select a.id,a.username from userinfo a, article bwhere a.id = b.author_id and b.type = 1;

登录后复制

记踩到 MySQL in 子查询的“坑”

五、补充

版本 5.6 已针对子查询做了优化,方式跟【四】中的临时表方式一样,参考官方文档:

If materialization is not used, the optimizer sometimes rewrites a noncorrelated subquery as a correlated subquery.

For example, the following IN subquery is noncorrelated  ( where_condition involves only columns from t2 and not t1 ):

select * from t1

where t1.a in (select t2.b from t2 where where_condition);

The optimizer might rewrite this as an EXISTS correlated subquery:

select * from t1

where exists (select t2.b from t2 where where_condition and t1.a=t2.b);

Subquery materialization using a temporary table avoids such rewrites and makes it possible to execute the subquery only once rather than once per row of the outer query.

https://dev.mysql.com/doc/refman/5.6/en/subquery-materialization.html

文章来自微信公众号:HULK一线技术杂谈

以上就是记踩到 MySQL in 子查询的“坑”的详细内容,更多请关注【创想鸟】其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至253000106@qq.com举报,一经查实,本站将立刻删除。

发布者:PHP中文网,转转请注明出处:https://www.chuangxiangniao.com/p/2201609.html

(0)
上一篇 2025年2月25日 23:41:20
下一篇 2025年2月25日 23:41:37

AD推荐 黄金广告位招租... 更多推荐

相关推荐

  • pycharm怎么导入库

    如何在 PyCharm 中导入库?手动导入:使用 import 语句导入整个库。使用 from…import 语句导入特定模块或函数。使用 Package Installer:通过菜单选项安装库。使用快捷键 (Ctrl + Al…

    2025年2月26日
    200
  • pycharm的debug怎么用

    使用 PyCharm 调试 Python 代码的步骤:设置断点启动调试会话单步执行代码检查变量评估表达式检查堆栈跟踪 PyCharm 调试用法 PyCharm 是一款流行的 Python IDE,它提供了强大的调试功能,帮助开发人员快速找出…

    2025年2月26日
    200
  • pycharm怎么导入模块

    在 PyCharm 中导入模块的方法:从 Python 解释器导入:添加模块到项目可用的模块列表中。使用快捷键:按住 “Ctrl” 或 “Cmd” 并键入模块名称,快速导入模块。使用 &#82…

    2025年2月26日
    200
  • pycharm怎么保存代码

    在 PyCharm 中保存代码的方法:按快捷键 Ctrl + S(Windows/Linux)、⌘ + S(macOS);点击菜单栏文件 (File) > 保存 (Save);点击文件 (File) > 另存为 (Save As…

    2025年2月26日
    200
  • pycharm怎么切换python版本

    PyCharm 中切换 Python 版本的步骤:打开设置并导航到“Python 解释器”。在“项目解释器”列表中选择所需版本,或单击“添加”并选择版本。单击“应用”按钮以应用更改,可能需要重新启动 PyCharm。 如何在 PyCharm…

    2025年2月26日
    200
  • pycharm怎么打断点

    使用 PyCharm 进行断点调试的方法包括:设置断点启动调试会话逐行运行代码检查变量值评估表达式继续执行提示:还可以在调试过程中使用断点条件、跳到光标和附加到进程等功能。 如何使用 PyCharm 断点调试 要使用 PyCharm 进行断…

    2025年2月26日
    200
  • pycharm怎么返回上一步

    PyCharm 返回上一步有两种方法:使用键盘快捷键:Windows/Linux:Ctrl + Z,macOS:Cmd + Z使用菜单:点击“编辑”菜单 > “撤销” PyCharm 中如何返回上一步? 在 PyCharm 中返回上一…

    2025年2月26日
    200
  • pycharm怎么替换字符

    PyCharm 中的字符替换功能可以帮助您快速地替换指定字符。只需打开查找和替换对话框 (Ctrl + F/Cmd + F),输入要查找和替换的字符。默认情况下,搜索将在当前文件或选定的文本中进行。高级选项包括使用正则表达式和替换模式,以便…

    2025年2月26日
    200
  • pycharm中怎么快捷注释

    在 PyCharm 中,可通过以下快捷键对代码进行注释:行内注释:Ctrl + / 或 Cmd + /块注释:Ctrl + Shift + / 或 Cmd + Shift + / PyCharm 中快捷注释的方法 在 PyCharm 中,有…

    2025年2月26日
    200
  • pycharm怎么改环境变量

    修改 PyCharm 的环境变量步骤:打开设置并导航到 “Project Interpreter”。在 “Environment Variables” 选项卡中,添加、编辑或删除变量。输入变量…

    2025年2月26日
    200

发表回复

登录后才能评论