热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

MysqlExists与in_在MySQL里,有个和in一样的东东叫做exists,但是它比in更牛叉,你会么?...

我们在学习Yii2的时候,一定接触过这样的where输入$query-where([exists,xxxx]);User::find()-where([ex

我们在学习Yii2的时候,一定接触过这样的where输入

$query->where(["exists",xxxx]);

User::find()->where(["exists",xxxx])->all();

是的,这是MYSQL的exists关键词,今天我们就来说说这个exist,为了给大家更清楚的讲解,先给大家说下本文目录:

什么是exists

exists和in的区别和使用场景

使用Yii2的Query Builder实现一个exists语句

要自己看哈。

提前准备

为了大家学习方便,北哥在这里面建立两张表并为其添加一些数据

10674ec76ebc4115163e5c34718248bf.png

一张会员表,一张会员下单表。

会员表数据

id

user

email

1

abei

abei@nai8.me

2

wh

abei@maige123.com

3

liuhuan

267765@qq.com

订单表

id

user_id

create_time

...

1

1

1489579802

...

2

2

1489579802

...

3

1

1489579802

...

4

3

1489579802

...

5

2

1489579802

...

6

1

1489579802

...

我们将用这两张表做演示。

什么是exists

exists表示存在,它常常和子查询配合使用,例如下面的SQL语句

SELECT * FROM `user`

WHERE exists (SELECT * FROM `order` WHERE user.id = order.user_id)

exists用于检查子查询是否至少会返回一行数据,该子查询实际上并不返回任何数据,而是返回值True或False。

当子查询返回为真时,则外层查询语句将进行查询。

当子查询返回为假时,外层查询语句将不进行查询或者查询不出任何记录。

因此上面的SQL语句旨在搜索出所有下过单的会员。需要注意的是,当我们的子查询为 SELECT NULL 时,MYSQL仍然认为它是True。

exists和in的区别和使用场景

是的,其实上面的例子,in这货也能完成,如下面SQL语句

SELECT * FROM `user`

WHERE id in (SELECT user_id FROM `order`)

那么!in和exists到底有啥区别那,要什么时候用in,什么时候用exists那?接下来阿北一一教你。

我们先记住口诀再说细节!“外层查询表小于子查询表,则用exists,外层查询表大于子查询表,则用in,如果外层和子查询表差不多,则爱用哪个用哪个。”

In关键字原理

SELECT * FROM `user`

WHERE id in (SELECT user_id FROM `order`)

in()语句只会执行一次,它查出order表中的所有user_id字段并且缓存起来,之后,检查user表的id是否和order表中的user_id相当,如果相等则加入结果期,直到遍历完user的所有记录。

in的查询过程类似于以下过程

$result = [];

$users = "SELECT * FROM `user`";

$orders = "SELECT user_id FROM `order`";

for($i &#61; 0;$i <$users.length;$i&#43;&#43;){

for($j &#61; 0;$j <$orders.length;$j&#43;&#43;){

// 此过程为内存操作&#xff0c;不涉及数据库查询。

if($users[$i].id &#61;&#61; $orders[$j].user_id){

$result[] &#61; $users[$i];

break;

}

}

}

我想你已经看出来了&#xff0c;当order表数据很大的时候不适合用in&#xff0c;因为它最多会将order表数据全部遍历一次。

如&#xff1a;user表有10000条记录,order表有1000000条记录,那么最多有可能遍历10000*1000000次,效率很差.

再如&#xff1a;user表有10000条记录,order表有100条记录,那么最多有可能遍历10000*100次,遍历次数大大减少,效率大大提升.

exists关键字原理

SELECT * FROM &#96;user&#96;

WHERE exists (SELECT * FROM &#96;order&#96; WHERE user.id &#61; order.user_id)

在这里&#xff0c;exists语句会执行user.length次&#xff0c;它并不会去缓存exists的结果集&#xff0c;因为这个结果集并不重要&#xff0c;你只需要返回真假即可。

exists的查询过程类似于以下过程

$result &#61; [];

$users &#61; "SELECT * FROM &#96;user&#96;";

for($i&#61;0;$i

if(exists($users[$i].id)){// 执行SELECT * FROM &#96;order&#96; WHERE user.id &#61; order.user_id

$result[] &#61; $users[$i];

}

}

你看到了吧&#xff0c;当order表比user表大很多的时候&#xff0c;使用exists是再恰当不过了&#xff0c;它没有那么多遍历操作,只需要再执行一次查询就行。

如:user表有10000条记录,order表有1000000条记录,那么exists()会执行10000次去判断user表中的id是否与order表中的user_id相等.

如:user表有10000条记录,order表有100000000条记录,那么exists()还是执行10000次,因为它只执行user.length次,可见B表数据越多,越适合exists()发挥效果.

但是&#xff1a;user表有10000条记录,order表有100条记录,那么exists()还是执行10000次,还不如使用in()遍历10000*100次,因为in()是在内存里遍历,而exists()需要查询数据库,我们都知道查询数据库所消耗的性能更高,而内存比较很快.

因此我们只需要记住口诀&#xff1a;“外层查询表小于子查询表&#xff0c;则用exists&#xff0c;外层查询表大于子查询表&#xff0c;则用in&#xff0c;如果外层和子查询表差不多&#xff0c;则爱用哪个用哪个。”

Yii2使用exists

我想我只需要写一个Query Builder的用法&#xff0c;其他你应该能举一反三了吧

$query &#61; new Query();

$query->from("user")

->where(["exists",(new Query())->select(&#39;user_id&#39;)->from(&#39;order&#39;)->where([&#39;user.id&#39; &#61;> &#39;order.user_id&#39;])]);



推荐阅读
  • 本文介绍了如何在MySQL中将零值替换为先前的非零值的方法,包括使用内联查询和更新查询。同时还提供了选择正确值的方法。 ... [详细]
  • 本文介绍了如何使用php限制数据库插入的条数并显示每次插入数据库之间的数据数目,以及避免重复提交的方法。同时还介绍了如何限制某一个数据库用户的并发连接数,以及设置数据库的连接数和连接超时时间的方法。最后提供了一些关于浏览器在线用户数和数据库连接数量比例的参考值。 ... [详细]
  • 本文介绍了Redis的基础数据结构string的应用场景,并以面试的形式进行问答讲解,帮助读者更好地理解和应用Redis。同时,描述了一位面试者的心理状态和面试官的行为。 ... [详细]
  • 本文由编程笔记小编整理,介绍了PHP中的MySQL函数库及其常用函数,包括mysql_connect、mysql_error、mysql_select_db、mysql_query、mysql_affected_row、mysql_close等。希望对读者有一定的参考价值。 ... [详细]
  • 本文介绍了通过mysql命令查看mysql的安装路径的方法,提供了相应的sql语句,并希望对读者有参考价值。 ... [详细]
  • ubuntu用sqoop将数据从hive导入mysql时,命令: ... [详细]
  • 如何在php中将mysql查询结果赋值给变量
    本文介绍了在php中将mysql查询结果赋值给变量的方法,包括从mysql表中查询count(学号)并赋值给一个变量,以及如何将sql中查询单条结果赋值给php页面的一个变量。同时还讨论了php调用mysql查询结果到变量的方法,并提供了示例代码。 ... [详细]
  • MyBatis多表查询与动态SQL使用
    本文介绍了MyBatis多表查询与动态SQL的使用方法,包括一对一查询和一对多查询。同时还介绍了动态SQL的使用,包括if标签、trim标签、where标签、set标签和foreach标签的用法。文章还提供了相关的配置信息和示例代码。 ... [详细]
  • 本文介绍了在CentOS上安装Python2.7.2的详细步骤,包括下载、解压、编译和安装等操作。同时提供了一些注意事项,以及测试安装是否成功的方法。 ... [详细]
  • 篇首语:本文由编程笔记#小编为大家整理,主要介绍了软件测试知识点之数据库压力测试方法小结相关的知识,希望对你有一定的参考价值。 ... [详细]
  • PreparedStatement防止SQL注入
    添加数据:packagecom.hyc.study03;importcom.hyc.study02.utils.JDBCUtils;importjava.sql ... [详细]
  • mysql自动打开文件_让docker中的mysql启动时自动执行sql文件
    本文提要本文目的不仅仅是创建一个MySQL的镜像,而是在其基础上再实现启动过程中自动导入数据及数据库用户的权限设置,并且在新创建出来的容器里自动启动My ... [详细]
  • 在数据分析工作中,我们通常会遇到这样的问题,一个业务部门由若干业务组构成,需要筛选出每个业务组里业绩前N名的业务员。这其实是一个分组排序的 ... [详细]
  • 前景:当UI一个查询条件为多项选择,或录入多个条件的时候,比如查询所有名称里面包含以下动态条件,需要模糊查询里面每一项时比如是这样一个数组条件:newstring[]{兴业银行, ... [详细]
  • Oracle seg,V$TEMPSEG_USAGE与Oracle排序的关系及使用方法
    本文介绍了Oracle seg,V$TEMPSEG_USAGE与Oracle排序之间的关系,V$TEMPSEG_USAGE是V_$SORT_USAGE的同义词,通过查询dba_objects和dba_synonyms视图可以了解到它们的详细信息。同时,还探讨了V$TEMPSEG_USAGE的使用方法。 ... [详细]
author-avatar
Maugham
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有