not in 和 in的陷阱

sql 在not in 子查询有null值情况下经常出现的陷阱


1、not in 和in  根据某个字段查询如 not in('3'),查不出这个字段为空的数据

2、not in 和 in 在 查询时,条件里不能存在null,如not in('',null,'3') 这样子是查询不出结果的

还有个陷阱是:

select count(1) from (MT_BDG_THREE1)  where uuid in('437044e9-660c-401b-ba22-b8e0d695e057')  and audit_state not in('3') 

当表里存在audit_state 为空的数据时,这些数据无法查出来

1、在创建数据时audit_state 赋予初始值

2、nvl函数为空值赋值,然后筛选



如果下:Table_A表和Table_B表,要求查询出在Table_A表中不在Table_B表中的记录。

复制代码
CREATE TABLE [dbo].[Table_A](
 [ID] [nchar](10) NULL,
 [Name] [nchar](10) NULL
) ON [PRIMARY]

GO

ID          Name
001        张三        
002        李四        
003        王五              

 

CREATE TABLE [dbo].[Table_B](
 [ID] [nchar](10) NULL,
 [Name] [nchar](10) NULL
) ON [PRIMARY]

GO

ID        Name
NULL     张三        
002       李四        
NULL     王五 
复制代码

 

很容大家第一时间相当的写法是:

SELECT  *
FROM    dbo.Table_A AS a
WHERE   a.ID NOT IN ( SELECT    b.ID
                      FROM      dbo.Table_B AS b)

然而查询出来并没有达到预期的

ID    Name
001  张三        
003  王五          

原因很简单:由于NULL不能进行如何的“操作”

–如果null参与算术运算,则该算术表达式的值为null。(例如:+,-,*,/ 加减乘除)

–如果null参与比较运算,则结果可视为false。(例如:>=,<=,<>  大于,小于,不等于)

–如果null参与聚集运算,则聚集函数都置为null。除count(*)之外。

--如果在not in子查询中有null值的时候,则不会返回数据。  (最简单的解释请参考下面的评论说明)     

复制代码
--正确写法                      
SELECT  *
FROM    dbo.Table_A AS a
WHERE   a.ID NOT IN ( SELECT    b.ID
                      FROM      dbo.Table_B AS b
                      WHERE     b.ID IS NOT NULL ) --排除NULL值参与运算符比较
                      
--建议修改为关联查询方法                                            
--正确写法1             
SELECT  *
FROM    dbo.Table_A AS a
WHERE   NOT EXISTS ( SELECT *
                     FROM   dbo.Table_B AS b
                     WHERE  a.ID = b.ID )
--正确写法2                     
SELECT  *
FROM    dbo.Table_A AS a
        LEFT OUTER JOIN dbo.Table_B AS b ON a.ID = b.ID
WHERE   b.ID IS NULL
复制代码
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值