mysql(4)—— 表连接查询与where后使用子查询的性能分析。

摘要:
子查询的子语句可以位于SQL语句的FROM、JOIN和WHERE之后。本文主要分析在WHERE之后使用子查询将查询与表连接的性能。对于表联接查询和子查询的性能,有不同的意见。一般认为,表联接查询的性能高于子查询。上述SQL语句的平均执行时间约为34秒,比使用IN快,但无法与表联接查询相比。

子查询就是在一条查询语句中还有其它的查询语句,主查询得到的结果依赖于子查询的结果。

子查询的子语句可以在一条sql语句的FROM,JOIN,和WHERE后面,本文主要针对在WHERE后面使用子查询与表连接查询的性能做出一点分析。

对于表连接查询和子查询性能的讨论众说纷纭,普遍认为的是表连接查询的性能要高于子查询。本文将从实验的角度,对这两种查询的性能做出验证,并就实验结果分析两种查询手段的执行流程对性能的影响。

首先准备两张表

1,访问日志表mm_log有150829条记录(相关sql文件已放在文章结尾的链接中)。

mysql(4)—— 表连接查询与where后使用子查询的性能分析。第1张

2,用户表mm_member有373条记录(相关sql文件已放在文章结尾的链接中)。

mysql(4)—— 表连接查询与where后使用子查询的性能分析。第2张

现在要求我们根据这两张表查出2017-02-06那天有那些用户登录过系统。

我们先来看一下使用表连接查询

SELECT
    SQL_NO_CACHE mm.*
FROM
    mm_member mm
JOIN
    mm_log ml
ON
    mm.id =ml.member_id
WHERE
    ml.access_time LIKE '%2017-02-06%'
GROUP BY
    ml.member_id;

这里使用了 SQL_NO_CACHE 是因为要多次执行这条sql语句,并计算出这条sql查询所耗费的平均时间,所以要关掉mysql的查询缓存,防止多次执行从缓存中读取数据。

mm.*是GROUP BYml.member_id分组后的诸多临时表的第一行数据,相关用法及原理请参见我的另一篇博客(http://www.cnblogs.com/cdf-opensource-007/p/6502556.html

对这条sql语句执行了10次,查询所耗费的平均时间在0.120s左右。

查询结果:(一共有5个用户访问过系统)

mysql(4)—— 表连接查询与where后使用子查询的性能分析。第3张

至于以上这条sql的执行流程已经在前几篇博客中描述的很详细了,这里就不再做叙述了。

下面使用WHERE后使用子查询的方式实现

SELECT
    SQL_NO_CACHE mm.username
FROM
    mm_member mm
WHERE
    mm.id IN(SELECT ml.member_id FROM mm_log ml WHERE ml.access_time LIKE '%2017-02-06%' GROUP BY ml.member_id);

当我第一次运行这条sql语句的时候,等了十几秒一直没有结果,我以为我的电脑死机,可Navicat显示处理中,最后40多秒的时候才运行出结果,接连运行了好多次在都是41秒左右出结果。

mysql(4)—— 表连接查询与where后使用子查询的性能分析。第4张

我们看到执行结果同上。那么使用子查询的性能到底低在哪里呢?

我的第一种推测是子语句:SELECT ml.member_id FROM mm_log ml WHERE ml.access_time LIKE '%2017-02-06%' GROUP BY ml.member_id耗费了大量的查询时间,因为mm_log这张表中有150829条记录。

把子语句单拿出来运行一下,发现子语句的运行时间也就在0.111s左右。

SELECT SQL_NO_CACHE member_id FROM mm_log ml WHERE ml.access_time LIKE '%2017-02-06%' GROUP BY ml.member_id;

这就说明我的第一种推测是不合理的。

那就分析下在WHERE后使用子查询IN的执行原理:

1,IN后面跟的子查询语句的执行结果只能有一列是用来和IN前面的主表的字段匹配的,在这里指的是mm.id。

2,一条带有子查询的sql语句首先执行的是子语句,主表的数据行按照IN前面主表的字段依次跟子查询的结果进行匹配,子查询中结果中有该数据行对应字段的值,则返回true,该行被WHERE筛选出来。没有则返回false,该行不被筛选。

3,那么按照2的说法,子查询的效率应该也不低啊,子语句的耗时在0.111s左右,而且主表mm_member和子语句的查询结果相匹配的次数,肯定是要少于表连接查询时数据行间匹配的次数的,但实验结果显示使用子查询的性能确实很低。

所以我有了第二种推测,主表mm_member数据行的每一行在与IN后面子语句的结果相匹配时,子语句都会重新执行一次,也就是说子语句第一次执行时,不会在内存中有缓存。这类似与使用了两个FOR循环嵌套,外层的FOR循环每拿出一个值,内层的FOR循环都要遍历一次。

那么根据以上的推测,拿主表mm_member的数据行数乘以子语句的执行时间就应该是整个查询的时间。

mm_member的数据行数:373

多次执行子语句算出平均时间在:0.111s

整个查询耗时的理论时间:41.403s

多次执行整个查询得出实际查询时间的平均值:40.834s

计算误差:(理论值-实际值)÷理论值 = 1.37%

误差还是在可以接受的范围内的,可以证明以上的推测。

根据以上的实验,我们可以得出的结论是,表连接查询的性能是要高于子查询的。

另外,对于在子查询中使用IN的性能高还是是用EXITS的性能高,有一种普遍的说法是:

1,在外表大,内表小,外表中有索引的情况下,使用IN。

2,在外表小,内表大,内表中有索引的情况下,使用EXITS

先介绍一下EXITS的用法,刚好本例符合外表小内表大的情况,就以本例介绍一下。看下SQL:

SELECT
    SQL_NO_CACHE mm.*
FROM
    mm_member mm
WHERE
    EXISTS(SELECT * FROM mm_log ml WHERE mm.id = ml.member_id AND ml.access_time LIKE '%2017-02-06%');

EXITS又简称代入查询,就是把主表的每一行代入子表的每一行进行检验,一旦子表中有符合的数据行就返回true,就可以取得主表中代入检验的那一行数据,否则返回false,不可以取得主表中代入检验的那一行数据。同IN不同的是,EXITS后的子语句不查询出结果,所以说SELECT后面的字段没有意义,一般使用*代替,由于EXITS的这种机制,当子表数据量比较大且有冗余字段的时候就很有可能避免了对子表的全表扫描,不像IN那样每次主表数据行来匹配都要进行全表扫描,并返回结果。所以说EXITS类似于两个FOR循环嵌套时,内层的FOR循环里面有 if(xxx){ break; }这种语法。

以上sql执行时间的平均在34秒左右,比使用IN要快上一些,但是跟表连接查询还不能比。

但是,在表与表之间没有关联关系时,就只能使用IN了。

sql 文件位置:http://pan.baidu.com/s/1gfLwIwr

最后说一点,我们作为程序员,研究问题还是要仔细深入一点的。当你对原理了解的有够透彻,开发起来也就得心应手了,很多开发中的问题和疑惑也就迎刃而解了,而且在面对其他问题的时候也可做到触类旁通。当然在开发中没有太多的时间让你去研究原理,开发中要以实现功能为前提,可等项目上线的后,你有大把的时间或者空余的时间,你大可去刨根问底,深入的去研究一项技术,为觉得这对一名程序员的成长是很重要的事情。

免责声明:文章转载自《mysql(4)—— 表连接查询与where后使用子查询的性能分析。》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇vivo统一告警平台设计与实践IOS 蓝牙(GameKit、Core Bluetooth)下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

mysql索引主键自增的测试

首先创建一张表 把主键设为自增,我们要测试这个主键到底是怎么自增这个id的 首先插入多条数据,id 不同, sql如下 INSERT INTO `hr`.`test` (`ad`) VALUES ('1'); INSERT INTO `hr`.`test` (`ad`) VALUES ('2'); INSERT INTO `hr`.`test` (`ad`...

mysql关于“异步复制”“同步复制”“半同步复制”“无损复制”的概念与区别

异步复制:主把事务写到binlog日志并不管从是否接收或者什么时候接收,commit之后,不会与从发生ack之类的交互。 同步复制:当主提交一个事务,在主向前端返回一个commit成功的指令前,必须保证所有的从已经提交了这个事务(所有从不但接收了,还必须apply了该事务日志) 半同步复制:(5.7.2版本之前默认且唯一的一个参数值after_commit...

SQL语句之语法汇总(一)

前段时间安装了sqlserver及management,编写了一些sql语句,现在对sql中常用的几个语法进行总结、分析与代码实例演示。 汇总一只介绍基本语法,较复杂的排序、分组等操作将在之后的文章中陆续总结! 一.创建表、修改表与删除表 1.1代码 1.创建表: create table Person5(Id int not null,Name nvar...

java部署ubuntu后中文显示问号问题

1、首先先回忆自身项目的编码格式,即在本地进行编码时使用的编码格式。UTF-82、检测tomcat的设置问题,在web.xml和server中的设置:server.xml中: <Connector port="8080" protocol="HTTP/1.1" connectionTimeout="20000" redirectPort="8443...

python selenium 基本常用操作

 最近学习UI自动化,把一些常用的方法总结一下,方便自己以后查阅需要。因本人水平有限,有不对之处多多包涵!欢迎指正! 一、xpath模糊匹配定位元素 武林至尊,宝刀屠龙刀(xpath),倚天不出(css),谁与争锋  学会了xpath,妈妈再也不用担心我定位不到元素啦 ^_^ # coding:utf-8 import time from seleniu...

数据库(二)

修改表格   如果SQL server 2008中无法修改表结构,提示错误为:不允许保存修改,…… 解决方案:工具→选项→左侧的Designers→表设计器和数据库设计器 去掉“阻止保存要求重新创建表的更改”前面的钩,重新启动系统。 --修改数据库的名字将student的名字修改成xueshengsp_renamedb student,xuesheng 增...