SQL子查询返回奇怪的结果

问题描述 投票:0回答:1

我正在MS SQL中编写一个非常简单的子查询,它会返回意外的结果。我想要返回的是一个存在于一个时间段但不存在于另一个时间段的acct数列表。我的初始查询(粘贴在下面)返回零行,这是意外的。

select
    name1.acct
from
    sym.dbo.name as name1
where
    name1.processdate = 20171130
    and name1.type = 0
    and name1.acct not in
    (
    select 
        name2.acct
    from 
        sym.dbo.name as name2
    where
        name2.type = 0
        and name2.processdate = 20171031
    )

但是,当我向子查询添加一行额外的逻辑并执行预期结果时,会返回给我,请参阅下文。

select
    name1.acct
from
    sym.dbo.name as name1
where
    name1.processdate = 20171130
    and name1.type = 0
    and name1.acct not in
    (
    select 
        name2.acct
    from 
        sym.dbo.name as name2
    where
        name2.type = 0
        and name2.processdate = 20171031
        and name2.acct <> '8888888'
    )

最初我认为它可能与数据类型有关,所以我尝试将acct编号转换为varchar和int,结果相同(使用convert而不是cast)。有人可以向我解释为什么查询一个返回零结果和查询两个返回预期结果?

我正在使用Microsoft SQL Management Studio 2016。

谢谢

sql sql-server subquery sql-server-2016
1个回答
3
投票

不要使用NOT IN,尤其是子查询。它的行为并不像预期的那样 - 正如您刚刚学到的那样。问题?如果子查询返回的任何值是NULL,那么NOT IN会过滤掉所有内容。

相反,使用NOT EXISTS

where name1.processdate = 20171130 and
      name1.type = 0 and
      not exists (select 1
                  from sym.dbo.name name2
                  where name2.acct = name1.acct and
                        name2.processdate = 20171031
                 );
© www.soinside.com 2019 - 2024. All rights reserved.