繁体   English   中英

SQL SELECT与m:n的关系

[英]SQL SELECT with m:n relationship

我在用户和标签之间有m:n的关系。 一个用户可以拥有m个标签,一个标签可以属于n个用户。 表看起来像这样:

USER:
ID
USER_NAME

USER_HAS_TAG:
USER_ID
TAG_ID

TAG:
ID
TAG_NAME

假设我需要选择所有标签为“apple”,“orange”和“banana”的用户。 使用SQL(MySQL DB)实现此目的的最有效方法是什么?

SELECT  u.*
FROM    (
        SELECT  user_id
        FROM    tag t
        JOIN    user_has_tag uht
        ON      uht.tag_id = t.id
        WHERE   tag_name IN ('apple', 'orange', 'banana')
        GROUP BY
                user_id
        HAVING  COUNT(*) = 3
        ) q
JOIN    user u
ON      u.id = q.user_id

通过删除HAVING COUNT(*) ,你得到OR而不是AND (尽管它不是最有效的方式)

通过更换32 ,你有精确定义二三标签的用户。

通过用>= 2替换= 3 ,您将获得至少定义了三个标签中的两个的用户。

除了其他好的答案之外,还可以检查WHERE子句中的条件:

select *
from user u
where 3 = (
    select count(distinct t.id)
    from user_has_tag uht
    inner join tag t on t.id = uht.tag_id
    where t.name in ('apple', 'orange', 'banana') 
    and uht.user_id = u.userid
)

count(distinct ...)确保标记只计算一次,即使用户有多个“香蕉”标记。

顺便说一句,该网站fruitoverflow.com尚未注册:)

你可以用连接完成所有操作......

select u.*
from user u

inner join user_has_tag ut1 on u.id = ut1.user_id 
inner join tag t1 on ut1.tag_id = t1.id and t1.tag_name = 'apple'

inner join user_has_tag ut2 on u.id = ut2.user_id 
inner join tag t2 on ut2.tag_id = t2.id and t2.tag_name = 'orange'

inner join user_has_tag ut3 on u.id = ut3.user_id 
inner join tag t3 on ut3.tag_id = t3.id and t3.tag_name = 'banana'
SELECT *
FROM USER u
INNER JOIN USER_HAS_TAG uht
ON u.id = uht.user_id
INNER JOIN TAG t
ON uht.TAG_ID = t.ID
WHERE t.TAG_NAME IN ('apple','orange','banana')

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM