说我有这样的查询:
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
这可行吗?还有更好的选择吗?
谢谢!
答案 0 :(得分:5)
使用:
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))
答案 1 :(得分:1)
你可以使用减号运算符。
伪查询
select everything
from tables
where id in ( select id from table minus select id from table where val is bad )
答案 2 :(得分:1)
你可以试试像
这样的东西SELECT *
FROM TABLE
WHERE TABLE.ID NOT IN(
SELECT ID
FROM TABLE
WHERE TABLE.VAL < '1'
OR TABLE.VAL > '3'
)
答案 3 :(得分:1)
这是另一种替代方案,它将通过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