【问题标题】:Exclude rows based on other rows (SQL)根据其他行排除行 (SQL)
【发布时间】:2011-01-27 23:57:09
【问题描述】:

假设我有这样的查询:

SELECT *
FROM TABLE

它返回这个:

TABLE
ID | DATA | VAL
===============
01 | ABCD | 1
01 | DEFG | 2
02 | FGHI | 3
02 | HIJK | 2
03 | JKLM | 3
03 | LMNO | 4
04 | NOPQ | 0
04 | PQRS | 1

目前我有一个查询尝试只找到像这样的好值,但它有缺陷,因为在其他行中包含错误 VAL 的 ID,这不是我想要的。

SELECT *
FROM TABLE
WHERE TABLE.VAL IN ("1","2","3")

会返回这个(缺少 LMNO 和 PQRS):

TABLE
ID | DATA | VAL
===============
01 | ABCD | 1
01 | DEFG | 2
02 | FGHI | 3
02 | HIJK | 2
03 | JKLM | 3
04 | NOPQ | 0

但是,我只想要 ID 没有错误值的行。所以,01 和 02 很好,因为它们的所有行都有很好的结果。 03 和 04 不好,因为它们被其他行中的坏结果污染了。

我可以像这样将结果带入并在软件中以这种方式处理它,但似乎这对数据库应该是可能的,并且作为一般规则,在数据库上执行比在软件中更好(你知道,这就是他们在那里的目的......)

我能想到的最好的是:

SELECT *
FROM TABLE
WHERE COUNT( SELECT ID
             FROM TABLE
             WHERE TABLE.VAL NOT IN ("1","2","3")
           ) = 0

这可行吗?有更好的选择吗?

谢谢!

【问题讨论】:

    标签: sql progress-db


    【解决方案1】:

    用途:

    SELECT * 
      FROM TABLE a
     WHERE a.val IN (1,2,3)
       AND NOT EXISTS(SELECT NULL
                        FROM TABLE b
                       WHERE b.id = a.id
                         AND b.val NOT IN (1, 2, 3))
    

    【讨论】:

    • 我相信这就是我所需要的。当然,我正在处理 Progress 4GL,他们尽可能地强调 SQL 标准,所以这给我带来了相当多的麻烦。
    • 作为记录,这确实解决了问题。结果这个家伙已经返回了 1000 多行,并在软件中删除它们以将其缩减到他需要的 80 左右。这将 24 秒的查询减少到 2(我敢打赌,其中大部分是事务开销......)谢谢!
    【解决方案2】:

    您可以使用减号运算符。

    伪查询

    select everything
    from tables
    where id in ( select id from table minus select id from table where val is bad )
    

    【讨论】:

    • MINUS 是 Oracle 特有的,不是吗? ANSI 除外
    【解决方案3】:

    你可以试试

    SELECT *
    FROM TABLE
    WHERE TABLE.ID NOT IN(
        SELECT ID
        FROM TABLE
        WHERE TABLE.VAL < '1'
        OR TABLE.VAL > '3'
    )
    

    【讨论】:

    • 我不能保证连续的数字:(
    • 好吧,您只需将子查询从大于/小于有效范围更改为 WHERE IN(错误值)
    【解决方案4】:

    这是另一种选择,它将通过 TBL 一次,聚合,并使用找到的 ID,从 TBL 中检索数据

    SELECT *
    WHERE ID IN
    (
        SELECT
           ID,
           CASE WHEN val in (1,2,3) THEN 1 ELSE 0 END Test
        FROM TBL
        GROUP BY ID
        HAVING MIN(val) = 1
    )
    

    对于多列键,作为上述 IN 形式的替代方案,您可以使用 JOIN 形式。

    SELECT T.*
    FROM (
        SELECT
           Company, OrderNumber,
           CASE WHEN val in (1,2,3) THEN 1 ELSE 0 END Test
        FROM TBL
        GROUP BY Company, OrderNumber
        HAVING MIN(val) = 1
        ) KEEP
    INNER JOIN TBL T ON T.Company = KEEP.Company and T.OrderNumber=KEEP.OrderNumber
    

    【讨论】:

    • 当ID是多键时有没有办法做到这一点?例如,我有一个公司和订单号来识别一条记录。
    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2020-08-12
    • 1970-01-01
    • 2019-05-21
    • 1970-01-01
    • 2021-07-15
    • 1970-01-01
    相关资源
    最近更新 更多