Обсуждение: Subselects and NOTs

Поиск
Список
Период
Сортировка

Subselects and NOTs

От
"Vadim B. Mikheev"
Дата:
Oracle 6, there is NULL into table b:

SQL> select * from a where x in (select * from b);

         X
----------
         2

SQL> select * from a where x not in (select * from b);

no rows selected

SQL> select * from a where not x in (select * from b);

no rows selected

Is 'not X in' the same as 'X not in' ? Currently we have:

vac=> select * from a where not x in (select * from b);
x
-
1
(1 row)

: subselect clause is "atomic" and NOT-s are never pushed into it.

Once again - what standard says ?

Vadim

Re: [HACKERS] Subselects and NOTs

От
Bruce Momjian
Дата:
>
> Oracle 6, there is NULL into table b:
>
> SQL> select * from a where x in (select * from b);
>
>          X
> ----------
>          2
>
> SQL> select * from a where x not in (select * from b);
>
> no rows selected
>
> SQL> select * from a where not x in (select * from b);
>
> no rows selected
>
> Is 'not X in' the same as 'X not in' ? Currently we have:

I am not sure about this, but I believe 'not X in subselect' is
evaluated as 'not (x in subselect)' and not as 'X not in subselect'.  Am
I missing something?

There is also some interesting stuff about comparisons:

( 1,2,NULL) = (3, NULL,4)   false
( 1,2,NULL) < (3, NULL,4)   true
( 1,2,NULL) = (1, NULL,4)   unknown
( 1,2,NULL) > (NULL, 2,4)   unknown

This happens because the comparisons are:

    left < right is true of and only if there exists some j such
that Lj < Rj is true and for all i < j, and Li = Ri is true

so it seems it compares these things from left to right, trying to make
the comparison.  For = and <>, is doesn't matter, but for the <, it does
matter.

Also they show:

select *
from test
where x <> (select y
        from test2)

When test2 returns no rows, the query returns no rows because the
subquery returns a single row of NULL values.

Hope this helps.  I can give more detail if you want it.


>
> vac=> select * from a where not x in (select * from b);
> x
> -
> 1
> (1 row)
>
> : subselect clause is "atomic" and NOT-s are never pushed into it.
>
> Once again - what standard says ?
>
> Vadim
>
>


--
Bruce Momjian
maillist@candle.pha.pa.us

Re: [HACKERS] Subselects and NOTs

От
"Vadim B. Mikheev"
Дата:
Bruce Momjian wrote:
>
> >
> > Is 'not X in' the same as 'X not in' ? Currently we have:
>
> I am not sure about this, but I believe 'not X in subselect' is
> evaluated as 'not (x in subselect)' and not as 'X not in subselect'.  Am
> I missing something?

Even 'not (x in subselect)' doesn't help in Oracle! This works just
as 'x not in subselect'.

Ok, if we don't know what standard say then could you test this
in Informix, etc. Don't forget put NULL into table in subselect
to get 'no rows selected' for 'x not in subselect'...

>
> Also they show:
>
> select *
> from test
> where x <> (select y
>             from test2)
>
> When test2 returns no rows, the query returns no rows because the
> subquery returns a single row of NULL values.

This is exactly how it works in postgres and Oracle.

Vadim

Re: [HACKERS] Subselects and NOTs

От
"Thomas G. Lockhart"
Дата:
> > > Is 'not X in' the same as 'X not in' ? Currently we have:
> > I am not sure about this, but I believe 'not X in subselect' is
> > evaluated as 'not (x in subselect)' and not as 'X not in subselect'.
>
> Even 'not (x in subselect)' doesn't help in Oracle! This works just
> as 'x not in subselect'.
> > Also they show:
> >
> > select *
> > from test
> > where x <> (select y
> >             from test2)
> >
> > When test2 returns no rows, the query returns no rows because the
> > subquery returns a single row of NULL values.

The SQL92 standard sez:

           1) If NOT is specified in a <boolean test>, then let BP be the
            contained <boolean primary> and let TV be the contained <truth
            value>. The <boolean test> is equivalent to:

            ( NOT ( BP IS TV ) )

However, "a NOT IN b" is not the same as "NOT (a IN b)", and my SQL book
points out that "IN" is shorthand for "=ANY" and "NOT IN" is shorthand for
"<>ALL". Also, my Date book sez:

            In general, an all-or-any condition evaluates to TRUE if and only
if
            the corresponding comparison condition without the ALL (or ANY,
            respectively) evaluates to TRUE for ALL (or ANY, respectively) of

            the rows in the table represented by the table expression.
            (NOTE: If that table is empty, the ALL conditions return TRUE,
the
           ANY conditions return FALSE).

So, it looks to me that

  WHERE x IN (SELECT y FROM empty_table)

evaluates to FALSE, and

  WHERE x NOT IN (SELECT y FROM empty_table)

evaluates to TRUE.

I looked through my two reference books and my online draft copy of the SQL92
standard, and could not find much mention of operator precendence, and even
less on "NOT" precendence. The only mention I could find was the statement

         ... Where
         the precedence of operators is determined by the Formats of this
         International Standard or by parentheses, those operators are ef-
         fectively applied in the order specified by that precedence. Where
         the precedence is not determined by the Formats or by parentheses,
         effective evaluation of expressions is generally performed from
         left to right. However, it is implementation-dependent whether ex-
         pressions are actually evaluated left to right, particularly when
         operands or operators might cause conditions to be raised or if
         the results of the expressions can be determined without completely
         evaluating all parts of the expression.

However, it wasn't clear to me whether the above statement was referring to
operators in expressions or to operators in the BNF notation used to define
the language.

Also, in a completely different place:

         Operations on numbers are performed according to the normal rules
         of arithmetic, within implementation-defined limits, except as
         provided for in Subclause 6.12, "<numeric value expression>".

I can't believe that the standard isn't more explicit somewhere about
operator precedence, or that it is strictly a left-to-right evaluation.

                                                       - Tom