2

我无法理解 Oracle 返回这个奇怪结果的原因。我认为代码真的很清楚。

我期望没有条件=(在 OUTER JOIN 条件中不为空)+(在 OUTER JOIN 条件中为空)

因为我将外部连接右表列中的 IS NULL / IS NOT NULL 解释为 EXISTS / NOT EXISTS 条件。

为什么我错了?

DESCRIPTION                          COUNT(1)
---------------------------------- ----------
No condition                             6403
is NOT null in OUTER JOIN cond           6403
is not null in where cond                6401
is null in OUTER JOIN cond               6247
is null in where cond                       2
proof flh_id_messaggio is not null          0
proof flh_stato is not null                 0


  SELECT 'is null in OUTER JOIN cond ' description, count(1)
    FROM    netatemp.TMP_BACKLOG_NOBILLING2013 t
         LEFT OUTER JOIN
            eni_flussi_hub c ON
            c.flh_id_messaggio = t.flh_id_messaggio  
            AND c.flh_stato is null   
   WHERE 1 = 1 
       And t.flh_stato = 'PA'
         AND t.OWNER = 'ETL' 
UNION
  SELECT 'is NOT null in OUTER JOIN cond ' description, count(1)
    FROM    netatemp.TMP_BACKLOG_NOBILLING2013 t
         LEFT OUTER JOIN
            eni_flussi_hub c ON
            c.flh_id_messaggio = t.flh_id_messaggio  
            AND c.flh_stato is not null   
   WHERE 1 = 1 
       And t.flh_stato = 'PA'
         AND t.OWNER = 'ETL'
UNION
  SELECT 'is null in where cond ' description, count(1)
    FROM    netatemp.TMP_BACKLOG_NOBILLING2013 t
         LEFT OUTER JOIN
            eni_flussi_hub c ON
            c.flh_id_messaggio = t.flh_id_messaggio                
   WHERE 1 = 1 
       And t.flh_stato = 'PA'
         AND t.OWNER = 'ETL'
         AND c.flh_stato is null 
UNION
  SELECT 'is not null in where cond ' description, count(1)
    FROM    netatemp.TMP_BACKLOG_NOBILLING2013 t
         LEFT OUTER JOIN
            eni_flussi_hub c ON
            c.flh_id_messaggio = t.flh_id_messaggio                
   WHERE 1 = 1 
       And t.flh_stato = 'PA'
         AND t.OWNER = 'ETL'
         AND c.flh_stato is not null 
UNION
  SELECT 'No condition' description, count(1)
    FROM    netatemp.TMP_BACKLOG_NOBILLING2013 t
         LEFT OUTER JOIN
            eni_flussi_hub c ON
            c.flh_id_messaggio = t.flh_id_messaggio                
   WHERE 1 = 1 
       And t.flh_stato = 'PA'
         AND t.OWNER = 'ETL'
UNION select 'proof flh_stato is not null' description, count(1)
from eni_flussi_hub
where flh_stato is null         
UNION select 'proof flh_id_messaggio is not null' description, count(1)
from eni_flussi_hub
where flh_id_messaggio is null  
4

1 回答 1

3

EXISTS/NOT EXISTS等效查询是通过将NULL条件放在WHERE子句中而不是在子句中来获得的OUTER JOIN。顺便说一句,这是我们从您的结果中观察到的:

No condition                             6403
is not null in where cond                6401
is null in where cond                       2

主表中的 2 行在连接表中没有对应的 ID。


当您将条件放在OUTER JOIN子句中时,您是在告诉 Oracle 将OUTER JOIN 您的主表从联接表中的行子集。

由于c.flh_stato永远不会为空,因此条件是多余的,我们得到与无条件查询相同的结果:

No condition                             6403
is NOT null in OUTER JOIN cond           6403

使用c.flh_stato IS NULLjoin子句中的条件,我们将主表连接到一个空的结果集,因此我们得到主表每一行的一行(我们用这个条件推导出主表有6247行):

is null in OUTER JOIN cond               6247
于 2013-06-19T14:22:28.103 回答