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

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

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值