NOT IN 和NOT EXISTS

登陆账户表logins
code name status
a admin N
b guest N
c member N
今天写了一个简单的NOT IN语句,结果跟预期大相径庭,百度之发现深坑一个,遂录之。
联系人表contacts
code name logins_code
d zhangsan a
e lisi b
f wangwu (null)

contacts表的logins_code连接logins表的code,联系人可以绑定一个登陆用户,也可以不绑定,我们需要查logins获得没有绑定的账户,sql语句:

SELECT * FROM logins WHERE code NOT IN(SELECT logins_code FROM contacts);

预期应该是获得 c|member|N 的记录,但结果是Empty set.

SELECT logins_code FROM contacts;

返回的结果是(a,b,null),继续前推:

SELECT * FROM logins WHERE code NOT IN ('a','b',NULL);
=
SELECT * FROM logins WHERE code<>'a' AND code<>'b' AND code<>NULL;

MySQL的IN并不完全等同于多个OR条件子句,MySQL是将IN()列表中的数据先排序后二分查找。

问题出现了,在SQL中 =,!=,~=,^=,<>NULL 比较的结果都是FALSE,与NULL值比较的时候必须用IS NULL ,IS NOT NULL。如果你数据库中的字段都是默认为NULL,那你就要小心了,如果对查询做了一个判断,例如我上面code 为f的记录

SELECT * FROM contacts where logins_code='';

还是什么都查不到,应该是logins_code IS NULL

所以如果你表字段默认为NULL在进行判断的时候一定要长点心了,NOT IN的坑算是对这个判断做了个包装。

最后还是通过EXISTS解决了问题,

SELECT * FROM logins t1 WHERE NOT EXISTS (SELECT * FROM contacts t2 WHERE t1.code=t2.logins_code);

我还没打算解释这个为什么可以,因为我看到一个巨绕的文章,正在研究,待议~

本文由金沙官网线上发布于数据库,转载请注明出处:NOT IN 和NOT EXISTS

您可能还会对下面的文章感兴趣: