繁体   English   中英

正确连接多个多对多表——MySQL查询

[英]Correctly join multiple many-to-many tables - MySQL query

一个看似通用的 SQL 查询真的让我一无所知。 这是这种情况。 我有 3 个通用表(这里是简化版本):

Movie

id | title
-----------------------
1  | Evil Dead
-----------------------
2  | Bohemian Rhapsody
....

Genre

id | title
-----------------------
1  | Horror
-----------------------
2  | Comedy
....


Rating

id | title
-----------------------
1  | PG-13
-----------------------
2  | R
....

和 2 个多对多表来连接它们:

Movie_Genre

movie_id | genre_id


Movie_Rating

movie_id | rating_id

最初的挑战是编写一个查询,它允许我获取属于多种类型的电影(例如恐怖喜剧或科幻动作片)。

谢天谢地,我能够在这里找到这个解决方案MySQL:选择连接表匹配所有值的记录

但是,获取属于多个多对多表的记录的正确选项是什么? 例如,评级为 R 的恐怖喜剧。 有没有办法在没有子查询(或只有一个)的情况下做到这一点?

一种方法使用相关子查询:

select m.*
from movies m
where (select count(*)
       from movie_genre mg
       where mg.movie_id = m.id
      ) > 1 and
      (select count(*)
       from movie_rating mr
       where mr.movie_id = m.id
      ) > 1 ;

使用movie_genre(movie_id)movie_rating(movie_id)上的索引,这可能具有相当合理的性能。

以上可能是最有效的方法。 但是,如果您想避免子查询,一种方法是:

select mg.movie_id
from movie_genres mg join
     movie_ratings mr
     on mg.movie_id = mr.movie_id
group by mg.movie_id
having count(distinct mg.genre_id) > 0 and
       count(distinct mr.genre_id) > 0;

join之前聚合比上述更有效:

select mg.movie_id
from (select movie_id
      from mg_genres
      group by movie_id
      having count(*) >= 2
     ) mg join
     (select movie_id
      from mg_ratings
      group by movie_id
      having count(*) >= 2
     ) mr
     on mg.movie_id = mr.movie_id;

尽管您声明要避免使用子查询,但具有讽刺意味的是,没有子查询的版本可能在这三个选项中性能最差。

例如评级为 R 的恐怖喜剧

您可以join所有表连接在一起,按电影聚合并使用HAVING子句进行过滤:

select m.id, m.title
from movies m
inner join movie_genre  mg on mg.movid_id = m.id
inner join genre g on g.id = mg.genre_id
inner join movie_rating mr on mr.movie_id = m.id
inner join rating r on r.id = mr.rating_id
group by m.id, m.title
having 
    max(r.title = 'R') = 1 
    and max(g.title = 'Horror') = 1
    and max(g.title = 'Comedy') = 1

您还可以使用几个exists条件以及相关子查询:

select m.*
from movie m
where
    exists (
        select 1 
        from movie_genre mg
        inner join genre g on g.id = mg.genre_id
        where mg.movie_id = m.id and g.title = 'R')
    and exists (
        select 1
        from movie_rating mr
        inner join rating r on r.id = mr.rating_id
        where mr.movie_id = m.id and r.title = 'Horror'
    )
    and exists (
        select 1
        from movie_rating mr
        inner join rating r on r.id = mr.rating_id
        where mr.movie_id = m.id and r.title = 'Comedy'
    )

暂无
暂无

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

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