本文章介绍一篇关于sql中IN与EXISTS,第一个查询使用 EXISTS 而第二个查询使用

  • 栏目:数据 时间:2020-04-15 22:35
<返回列表

本文章介绍一篇关于sql中IN与EXISTS,NOT IN与NOT EXISTS的详细介绍,有需要了解的同学可以参考一下。

有一个查询如下: 复制代码 代码如下: SELECT c.CustomerId, CompanyName FROM Customers c WHERE EXISTS( SELECT OrderID FROM Orders o WHERE o.CustomerID = cu.CustomerID) 这里面的EXISTS是如何运作呢?子查询返回的是OrderId字段,可是外面的查询要找的是CustomerID和CompanyName字段,这两个字段肯定不在OrderID里面啊,这是如何匹配的呢? EXISTS用于检查子查询是否至少会返回一行数据,该子查询实际上并不返回任何数据,而是返回值True或False。 EXISTS 指定一个子查询,检测行的存在。语法:EXISTS subquery。参数 subquery 是一个受限的 SELECT 语句 。结果类型为 Boolean,如果子查询包含行,则返回 TRUE。 在子查询中使用 NULL 仍然返回结果集 这个例子在子查询中指定 NULL,并返回结果集,通过使用 EXISTS 仍取值为 TRUE。 复制代码 代码如下: SELECT CategoryName FROM Categories WHERE EXISTS (SELECT NULL) ORDER BY CategoryName ASC 比较使用 EXISTS 和 IN 的查询 这个例子比较了两个语义类似的查询。第一个查询使用 EXISTS 而第二个查询使用 IN。注意两个查询返回相同的信息。 复制代码 代码如下: SELECT DISTINCT pub_name FROM publishers WHERE EXISTS (SELECT * FROM titles WHERE pub_id = publishers.pub_id AND type = 'business') 复制代码 代码如下: SELECT distinct pub_name FROM publishers WHERE pub_id IN (SELECT pub_id FROM titles WHERE type = 'business') 比较使用 EXISTS 和 = ANY 的查询 本示例显示查找与出版商住在同一城市中的作者的两种查询方法:第一种方法使用 = ANY,第二种方法使用 EXISTS。注意这两种方法返回相同的信息。 复制代码 代码如下: SELECT au_lname, au_fname FROM authors WHERE exists (SELECT * FROM publishers WHERE authors.city = publishers.city) 复制代码 代码如下: SELECT au_lname, au_fname FROM authors WHERE city = ANY (SELECT city FROM publishers) 比较使用 EXISTS 和 IN 的查询 本示例所示查询查找由位于以字母 B 开头的城市中的任一出版商出版的书名: 复制代码 代码如下: SELECT title FROM titles WHERE EXISTS (SELECT * FROM publishers WHERE pub_id = titles.pub_id AND city LIKE 'B%') 复制代码 代码如下: SELECT title FROM titles WHERE pub_id IN (SELECT pub_id FROM publishers WHERE city LIKE 'B%') 使用 NOT EXISTS NOT EXISTS 的作用与 EXISTS 正相反。如果子查询没有返回行,则满足 NOT EXISTS 中的 WHERE 子句。本示例查找不出版商业书籍的出版商的名称: 复制代码 代码如下: SELECT pub_name FROM publishers WHERE NOT EXISTS (SELECT * FROM titles WHERE pub_id = publishers.pub_id AND type = 'business') ORDER BY pub_name 又比如以下 SQL 语句: 复制代码 代码如下: select distinct 姓名 from xs where not exists ( select * from kc where not exists ( select * from xs_kc where 学号=xs.学号 and 课程号=kc.课程号 ) 把最外层的查询xs里的数据一行一行的做里层的子查询。 中间的 exists 语句只做出对上一层的返回 true 或 false,因为查询的条件都在 where 学号=xs.学号 and 课程号=kc.课程号这句话里。每一个 exists 都会有一行值。它只是告诉一层,最外层的查询条件在这里成立或都不成立,返回的时候值也一样回返回上去。直到最高层的时候如果是 true就返回到结果集。为 false丢弃。 复制代码 代码如下: where not exists select * from xs_kc where 学号=xs.学号 and 课程号=kc.课程号 这个 exists 就是告诉上一层,这一行语句在我这里不成立。因为他不是最高层,所以还要继续向上返回。 select distinct 姓名 from xs where not exists ,由于是最高层所以就会把这行的结果返回到结果集。 几个重要的点: 最里层要用到的醒询条件的表比如:xs.学号、kc.课程号等都要在前面的时候说明一下select * from kc,select distinct 姓名 from xs 不要在太注意中间的exists语句. 把exists和not exists嵌套时的返回值弄明白

强调下:在含有NULL值的列的时候,就要小心了,not exists与not in在逻辑上是不等价的--先创建2张表

代码如下复制代码

use master;

if db_id('DbTest') is not null drop database DbTest;

create database DbTest;go

use DbTest;go

--创建Customers表create table Customers( custid INT NOT NULL IDENTITY, companyname NVARCHAR(40) NOT NULL, country NVARCHAR(15) NOT NULL, constraint pk_customer primary key(custid));

--创建Orders表

代码如下复制代码

CREATE TABLE Orders( orderid INT NOT NULL IDENTITY, custid INT NULL, CONSTRAINT PK_Orders PRIMARY KEY(orderid), CONSTRAINT FK_Orders_Customers FOREIGN KEY(custid) REFERENCES Customers(custid),); set identity_insert Customers on;

INSERT INTO Customers(custid, companyname,country) VALUES(1, N'大众', N'中国'); INSERT INTO Customers(custid, companyname,country) VALUES(2, N'宝马', N'美国'); INSERT INTO Customers(custid, companyname,country) VALUES(3, N'奔驰', N'中国'); INSERT INTO Customers(custid, companyname,country) VALUES(4, N'奇瑞', N'德国'); INSERT INTO Customers(custid, companyname,country) VALUES(5, N'福特', N'美国'); set identity_insert Customers off; set identity_insert Orders on;--custid代表员工号 INSERT INTO Orders(orderid, custid) VALUES(1,1); INSERT INTO Orders(orderid, custid) VALUES(2,2); INSERT INTO Orders(orderid, custid) VALUES(3,3); INSERT INTO Orders(orderid, custid) VALUES(4,4); INSERT INTO Orders(orderid, custid) VALUES(5,5);--查看表的数据select custid,companyname,country from Customers;select orderid,custid from Orders;--插入数据成功

--咱们回到正题,比较Exists与in,not exists与 not in

--查询来自中国,而且下过订单的所有客户

代码如下复制代码 select custid,companynamefrom Customers as Cwhere country=N'中国'and exists (select * from Orders as O where O.custid=C.custid);--返回--custid companyname--1 大众--3 奔驰

--外部查询返回来自中国的客户信息,对于这个客户,exists谓词在Orders表查找是否至少存在一个与外部客户行信息相同的custid订单行

--用IN查询刚刚的需求

代码如下复制代码 select custid,companynamefrom Customers as Cwhere country=N'中国'and custid in(select custid from Orders);

--结果跟上面的返回一样的值

--下面的知识点我们需要认识到:--当列表中有NULL时,in实际会产生一个UNKNOWN的结果,例如 a in(d,b,null)的结果是UNKNOWN,而a not in (d,b,null)返回的是not unknowd仍然是unknowd--而not in与not exists则结果会很不同,例如a in(a,b,null)返回的是TRUE,而a not in(a,b,null)返回的肯定是not true即为false--有了上面的认识,好继续开工了....--我们现在向Orders表插入一行数据

代码如下复制代码

set identity_insert Orders on;

insert into Orders(orderid,custid) values(6,null);

set identity_insert Orders off;

set identity_insert Customers on;

insert into Customers(custid,companyname,country) values(7,N'雷克萨斯',N'美国');

set identity_insert Customers off;

select * from Orders;select * from Customers;

--假设现在要返回来自美国且没有订单的客户

代码如下复制代码 select custid,companynamefrom Customers as Cwhere country=N'美国'and not exists (select * from Orders as O where O.custid=C.custid );--返回--custid companyname--7 雷克萨斯

--我们再用IN方法

代码如下复制代码 select custid,companyname from Customers as Cwhere country=N'美国'and custid not in(select custid from Orders);

--返回的结果为空!!!--为什么呢??--因为还记得我们刚插入的一行数据中custid为null么,这就是问题所在--not in (select custid from Orders)返回的实际是unknown,所以返回结果集为空,除非你显示的规定custid不能为空

--下面是正确的解决方法

代码如下复制代码 select custid,companynamefrom Customers as Cwhere country=N'美国'and custid not in (select custid from Orders where custid is not null);--返回--custid companyname--7 雷克萨斯

--所以在含有NULL值的列的时候,就要小心了,not exists与not in在逻辑上是不等价的

in适合内外表都很大的情况,exists适合外表结果集很小的情况。exists 和 in 使用一例 =========================================================== 今天市场报告有个sql及慢,运行需要20多分钟,如下:

代码如下复制代码 update p_container_decl cdset cd.ANNUL_FLAG='0001',ANNUL_DATE = sysdatewhere exists(select 1from (select tc.decl_no,tc.goods_nofrom p_transfer_cont tc,P_AFFIRM_DO adwhere tc.GOODS_DECL_NO = ad.DECL_NOand ad.DECL_NO = 'sssssssssssssssss') awhere a.decl_no = cd.decl_noand a.goods_no = cd.goods_no)

上面涉及的3个表的记录数都不小,均在百万左右。根据这种情况,我想到了前不久看的tom的一篇文章,说的是exists和in的区别,in 是把外表和那表作hash join,而exists是对外表作loop,每次loop再对那表进行查询。这样的话,in适合内外表都很大的情况,exists适合外表结果集很小的情况。

而我目前的情况适合用in来作查询,于是我改写了sql,如下:

代码如下复制代码 update p_container_decl cdset cd.ANNUL_FLAG='0001',ANNUL_DATE = sysdatewhere (decl_no,goods_no) in(select tc.decl_no,tc.goods_nofrom p_transfer_cont tc,P_AFFIRM_DO adwhere tc.GOODS_DECL_NO = ad.DECL_NOand ad.DECL_NO = ssssssssssss)

让市场人员测试,结果运行时间在1分钟内。问题解决了,看来exists和in确实是要根据表的数据量来决定使用。

请注意not in 逻辑上不完全等同于not exists,如果你误用了not in,小心你的程序存在致命的BUG:

请看下面的例子:

代码如下复制代码

create table t1 (c1 number,c2 number);create table t2 (c1 number,c2 number);

insert into t1 values (1,2);insert into t1 values (1,3);insert into t2 values (1,2);insert into t2 values (1,null);

select * from t1 where c2 not in (select c2 from t2);no rows foundselect * from t1 where not exists (select 1 from t2 where t1.c2=t2.c2);c1 c21 3

正如所看到的,not in 出现了不期望的结果集,存在逻辑错误。如果看一下上述两个select语句的执行计划,也会不同。后者使用了hash_aj。因此,请尽量不要使用not in(它会调用子查询),而尽量使用not exists(它会调用关联子查询)。如果子查询中返回的任意一条记录含有空值,则查询将不返回任何记录,正如上面例子所示。除非子查询字段有非空限制,这时可以使用not in ,并且也可以通过提示让它使用hasg_aj或merge_aj连接。

接着看

SQL中IN,NOT IN,EXISTS,NOT EXISTS的用法和差别: IN:确定给定的值是否与子查询或列表中的值相匹配。 IN 关键字使您得以选择与列表中的任意一个值匹配的行。 当要获得居住在 California、Indiana 或 Maryland 州的所有作者的姓名和州的列表时,就需要下列查询:

代码如下复制代码 SELECT ProductID, ProductName FROM Northwind.dbo.Products WHERE CategoryID = 1 OR CategoryID = 4 OR CategoryID = 5

然而,如果使用 IN,少键入一些字符也可以得到同样的结果:

代码如下复制代码 SELECT ProductID, ProductName FROM Northwind.dbo.Products WHERE CategoryID IN (1, 4, 5)

IN 关键字之后的项目必须用逗号隔开,并且括在括号中。 下列查询在 titleauthor 表中查找在任一种书中得到的版税少于 50% 的所有作者的 au_id,然后从 authors 表中选择 au_id 与 titleauthor 查询结果匹配的所有作者的姓名:

代码如下复制代码 SELECT au_lname, au_fname FROM authors WHERE au_id IN (SELECT au_id FROM titleauthor WHERE royaltyper 50)

结果显示有一些作者属于少于 50% 的一类。 NOT IN:通过 NOT IN 关键字引入的子查询也返回一列零值或更多值。 以下查询查找没有出版过商业书籍的出版商的名称。

代码如下复制代码 SELECT pub_name FROM publishers WHERE pub_id NOT IN (SELECT pub_韦德体育 ,id FROM titles WHERE type = 'business')

使用 EXISTS 和 NOT EXISTS 引入的子查询可用于两种集合原理的操作:交集与差集。两个集合的交集包含同时属于两个原集合的所有元素。 差集包含只属于两个集合中的第一个集合的元素。 EXISTS:指定一个子查询,检测行的存在。 本示例所示查询查找由位于以字母 B 开头的城市中的任一出版商出版的书名:

代码如下复制代码 SELECT DISTINCT pub_name FROM publishers WHERE EXISTS (SELECT * FROM titles WHERE pub_id = publishers.pub_id AND type = 'business') SELECT distinct pub_name FROM publishers WHERE pub_id IN (SELECT pub_id FROM titles WHERE type = 'business')

两者的区别: EXISTS:后面可以是整句的查询语句如:SELECT * FROM titles IN:后面只能是对单列:SELECT pub_id FROM titles NOT EXISTS: 例如,要查找不出版商业书籍的出版商的名称:

代码如下复制代码 SELECT pub_name FROM publishers WHERE NOT EXISTS (SELECT * FROM titles WHERE pub_id = publishers.pub_id AND type = 'business')

下面的查询查找已经不销售的书的名称:

代码如下复制代码 SELECT title FROM titles WHERE NOT EXISTS (SELECT title_id FROM sales WHERE title_id = titles.title_id)

上一篇:参照完整性(完整性约束)是数据库设计中的一个重要概念,CSDN Oracle技术论坛 下一篇:没有了

更多阅读

本文章介绍一篇关于sql中IN与EXISTS,第一

数据 2020-04-15
本文章介绍一篇关于sql中IN与EXISTS,NOT IN与NOTEXISTS的详细介绍,有需要了解的同学可以参考一下...
查看全文

参照完整性(完整性约束)是数据库设计中

数据 2020-04-15
问题来源:CSDN Oracle技术论坛 转自: 问题: myepoch提出相同SQL因为检索的值,不同执行效率差...
查看全文

一、创建新库

数据 2020-04-15
sql server里连接远程服务器,并进行创建和删除新数据库教程 CREATE PROCEDURE [dbo].[p_CreateDB]   @D...
查看全文

友情链接: 网站地图

Copyright © 2015-2019 http://www.koi-bumi.com. 韦德体育有限公司 版权所有