not in 和not exist

今天写了一个简单的NOT IN语句,结果跟预期大相径庭,百度之发现深坑一个,遂录之。

登陆账户表logins

codenamestatus
aadminN
bguestN
cmemberN

联系人表contacts

codenamelogins_code
dzhangsana
elisib
fwangwu(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;

PS: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);

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

    原文作者:Donne
    原文地址: https://segmentfault.com/a/1190000007737920
    本文转自网络文章,转载此文章仅为分享知识,如有侵权,请联系博主进行删除。
点赞