理解NULL如何影响IN和EXITS语句

本文探讨了SQL中IN和EXISTS语句的区别,尤其是在处理NULL值时的行为差异。介绍了这两种语句的基本用法,以及如何正确使用以避免常见的问题。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

从表面上看,IN和EXITS的SQL语句是可互换和等效的。然而,它们在处理UULL数据时会有很大的差别,并导致不同的结果。问题的根源是在一 个Oracle数据库中,一个NULL值意味着未知变量,所以操作NULL值的比较函数的结果也是一个未知变量,而且任何返回NULL的值通常也被忽略。 例如,以下查询都不会返回一行的值:

select 'true' from dual where 1 = null;

select 'true' from dual where 1 != null;

 

只有IS NULL才能返回true,并返回一行:

select 'true' from dual where 1 is null;

select 'true' from dual where null is null;

 

当你选择使用IN,你将会告诉SQL选择一个值并与其它每一值相比较。如果NULL值存在,将不会返回一行,即使两个都为NULL。

select 'true' from dual where null in (null);

select 'true' from dual where (null,null) in ((null,null));

select 'true' from dual where (1,null) in ((1,null));

一个IN语句在功能上相当于= ANY语句:

select 'true' from dual where null = ANY (null);

select 'true' from dual where (null,null) = ANY ((null,null));

select 'true' from dual where (1,null) = ANY ((1,null));

 

当你使用一个EXISTS等效形式的语句,SQL将会计算所有行,并忽略子查询中的值。

select 'true' from dual where exists (select null from dual);

select 'true' from dual where exists (select 0 from dual where null is null);

 

IN和EXISTS在逻辑上是相同的。IN语句比较由子查询返回的值,并在输出查询中过滤某些行。EXISTS语句比较行的值,并在子查询中过滤某些行。对于NULL值的情况,行的结果是相同的。

selectename from emp where empno in (select mgr from emp);

selectename from emp e where exists (select 0 from emp where mgr = e.empno);

 

然而当逻辑被逆向使用,即NOT IN 及NOT EXISTS时,问题就会产生:

selectename from emp where empno not in (select mgr from emp);

selectename from emp e where not exists (select 0 from emp where mgr =

e.empno);

 

NOT IN语句实质上等同于使用=比较每一值,如果测试为FALSE或者NULL,结果为比较失败。例如:

select 'true' from dual where 1 not in (null,2);

select 'true' from dual where 1 != null and 1 != 2;

select 'true' from dual where (1,2) not in ((2,3),(2,null));

select 'true' from dual where (1,null) not in ((1,2),(2,3));

这些查询不会返回任何一行。第二个查询语句更为明显,即1 != null,所以整个WHERE都为false。然而这些查询语句可变为:

select 'true' from dual where 1 not in (2,3);

select 'true' from dual where 1 != 2 and 1 != 3;

 

你也可以使用NOT IN查询,只要你保证返回的值不会出现NULL值:

selectename from emp where empno not in (select mgr from emp where mgr is not

null);

selectename from emp where empno not in (select nvl(mgr,0) from emp);

 

通过理解IN,EXISTS, NOT IN,以及NOT EXISTS之间的差别,当NULL出现在任一子查询中时,你可以避免一些常见的问题。

 

原文地址:http://www.kuqin.com/article/17mysql/2005041812275611291.html

使用Java写一个(一)系统功能说明 1、 用户管理系统主要功能:普通用户管理员用户功能。 2、 普通用户:可以查看全部用户信息,按用户年龄查看个人信息、修改自己的信息;可以尝试增加按照名字查看个人信息。 3、 管理员:删除用户、增加用户、修改用户、按ID查看用户、查看全部信息。可以尝试增加按照名字查看个人信息。 4、 用户的实体类的特征:用户ID编号、用户名、用户密码、家庭地址、年龄、电话等 (二)实现说明: 1.定义User类,用户ID编号、用户名、用户密码、家庭地址、年龄、电话、是否管理员。定义有参无参的两个构造方法,定义toString()方法方便打印对象信息。 2.系统中定义UserDAO类,定义密码错误次数属性成员;定义以下成员方法:查看全部用户信息、按用户年龄查看个人信息、按名字查看用户信息、按照ID查看信息、根据用户名以及密码修改用户的信息、根据用户编号修改信息、删除用户、增加用户、根据用户名密码登陆的功能。 3、模拟数据存储使用类Memoery中的userlist数组实现。该类中定义长度为100的User类型的静态数据成员,用于存放数据信息;定义用于描述数组中元素个数的静态数据成员;为了模拟实现数据存储,定义不含参数的构造函数,实现通过定义若干用户对象初始化数组成员中的一部分,其余成员也初始化为user对象,仅为对象是否管理员赋值为false。 4.定义Test类,在main方法中实现功能操作。要注意能循环操作,即输入用户名密码一次,能进行多种功能。使用循环语句实现。
06-11
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值