具有 IN 和 NOT IN 条件的单个表的内部查询

Posted

技术标签:

【中文标题】具有 IN 和 NOT IN 条件的单个表的内部查询【英文标题】:Inner queries on a single table with IN and NOT IN conditions 【发布时间】:2012-08-29 00:34:58 【问题描述】:

这是对我之前answered question的修改

我在下表中有数据:

ROLE_ID | USER_ID
------------------
 14     | USER A
 15     | USER A
 11     | USER B
 13     | USER D
 13     | USER A
 15     | USER B
 15     | USER D
 12     | USER C
 15     | USER C

我想获取只有 13 和 15 的用户 ID。所以根据上面的示例,我应该只返回 USER D

下面的查询是在我之前的回答中提供的,NOT IN 部分是我添加的,但是,这并没有达到目标..

select user_id
  from my_table
 where role_id in (13,15) AND role_id not in (11,14)
 group by user_id.
having count(distinct role_id) = 2

【问题讨论】:

您是否也希望拥有 role_id 13 15(仅此而已)的用户? 我想要角色 ID 为 13 和 15 的用户(仅此而已) 【参考方案1】:

只获得 13 和 15,请执行以下操作:

select user_id
from my_table
group by user_id
having max(case when role_id = 13 then 1 else 0 end) = 1 and  -- has 13
       max(case when role_id = 15 then 1 else 0 end) = 1 and  -- has 15
       max(case when role_id not in (13, 15) then 1 else 0 end) = 0 -- nothing else

这会检查 13 和 15 是否在 user_id 集中。然后它会检查集合中是否没有其他内容。

我意识到在 case 语句中使用 having 子句一开始似乎很尴尬。但是,您可以表达关于集合中事物的不同组合的大量逻辑。

【讨论】:

【参考方案2】:

假设user_idrole_id 的组合是唯一的,你可以这样做

select user_id
  from my_table
 where role_id in (13,15,11,14) 
 group by user_id.
having sum( case when role_id in (13,15) then 1 else 0 end) = 2
   and sum( case when role_id in (11,14) then 1 else 0 end) = 0

如果user_idrole_id 的组合不是唯一的,那么原始count 中的distinct 是必要的,并且事情变得更具挑战性。

【讨论】:

【参考方案3】:

我想要角色 ID 为 13 和 15(仅此而已)的用户

一种方法:

SELECT t13.user_id
FROM   my_table t13
JOIN   my_table t15 ON t13.user_id = t15.user_id
LEFT   JOIN my_table tx ON tx.user_id = t13.user_id AND tx.role_id NOT IN (13,15)
WHERE  t13.role_id = 13
AND    t15.role_id = 15
AND    tx.user_id IS NULL;

这假定 (role_id, user_id) 是唯一的。

这是关系除法的特例。我们最近收集了很多方法来处理这个问题(对于 PostgreSQL 和 mysql,但其中大部分是同样适用于 Oracle 的基本 SQL):How to filter SQL results in a has-many-through relation

另一个不存在的变体:

SELECT t13.user_id
FROM   my_table t13
JOIN   my_table t15 ON t13.user_id = t15.user_id
WHERE  t13.role_id = 13
AND    t15.role_id = 15
AND    NOT EXISTS (
   SELECT 1
   FROM   my_table tx
   WHERE  tx.user_id = t13.user_id
   AND    tx.role_id NOT IN (13,15)
   );

【讨论】:

【参考方案4】:

实际上并没有测试它,但是这样的东西应该可以工作:

SELECT USER_ID
FROM MY_TABLE T1
WHERE
    ROLE_ID IN (13, 15)
    AND NOT EXISTS (
        SELECT *
        FROM MY_TABLE T2
        WHERE
            T1.USER_ID = T2.USER_ID
            AND ROLE_ID IN (11, 14)
    )
GROUP BY USER_ID
HAVING COUNT(DISTINCT ROLE_ID) = 2

简而言之:忽略用户的所有行,其中任何行包含不需要的 ROLE_ID (NOT EXISTS),然后继续确保用户具有所有需要的 ROLE_ID。

注意:如果 ROLE_ID, USER_ID 上有键,则不需要 DISTINCT。


若要获取只有 (13, 15) 且没有其他内容的用户,并且事先不知道什么是“else”,只需将内部查询中的 ROLE_ID IN (11, 14) 替换为 ROLE_ID NOT IN (13, 15)

【讨论】:

【参考方案5】:

将我的answer 扩展到您原来的问题,您可以尝试

(SELECT user_id FROM table WHERE role_id=13
       INTERSECT
SELECT user_id FROM table WHERE role_id=15)
       MINUS
SELECT user_id FROM table WHERE role_id IN (11, 14)

INTERSECTMINUS 都特定于 ORACLE,因此其他解决方案更通用,而这更具可读性和更易于修改。

无权访问 ORACLE 数据库,因此未经过测试。

【讨论】:

以上是关于具有 IN 和 NOT IN 条件的单个表的内部查询的主要内容,如果未能解决你的问题,请参考以下文章

HasManyThrough 具有多态和多对多关系

html 文件上传图像具有最大大小和文件限制

Redshift 获取具有最高优先级和时间戳的行

如何使 TextArea 具有最大尺寸和滚动条?

当用户在 Blazor Webassembly 身份验证和授权中具有多个角色时出现问题?

XML 和 XSD 验证失败:元素同时具有“类型”属性和“匿名类型”子项