mysql使用LEFT JOIN从多个表中复用多个COUNT()

时间:2014-07-13 21:44:07

标签: php mysql

我想展示所有用户的结论。

我有3张桌子。

表格

post_id(index)     user_id
1                  1
2                  3
3                  3
4                  4

桌面照片

photo_id(index)     user_id
1                   2
2                   4
3                   1
4                   1

表格视频

photo_id(index)     user_id
1                   4
2                   4
3                   3
4                   3

并在表格用户

user_id(index)     user_name
1                  mark
2                  tommy
3                  john
4                  james

实际上,每个表都有超过4行。

我想要这样的结果。

id     name      post    photo   videos
1      mark      1       2       0
2      tommy     0       1       0
3      john      2       0       2
4      james     1       1       2
5      ..        ..      ..      ..

下面的代码是可以正常工作但速度非常慢的SQL,如果您帮助我使用LEFT JOIN如何帮助我,我将非常感激。谢谢。

SQL

"select user.*, 
(select count(*) from post where post.userid = user.userid) postCount,
(select count(*) from photo where photo.userid = user.userid) photoCount,
(select count(*) from video where video .userid = user.userid) videoCount
from user order by user.id"

(或ORDER BY postCount,photoCount或videoCount ASC或DESC,如我所愿)

我之前做过研究,但没有任何帮助过我。

3 个答案:

答案 0 :(得分:4)

你这样做的方法很合理。这是您的查询:

select user.*, 
       (select count(*) from post where post.userid = user.userid) as postCount,
       (select count(*) from photo where photo.userid = user.userid) as photoCount,
       (select count(*) from video where video.userid = user.userid) as videoCount
from user
order by user.id;

对于此查询,您需要以下索引:

post(userid)
photo(userid)
video(userid)
user(id)

您可能已经拥有了最后一个,因为user.id可能是该表的主键。

请注意,在这种情况下,left join方法不是一个好主意。这三个表 - 帖子,照片和视频 - 彼此独立。如果用户有五个,那么将它们连接在一起将产生125个中间行。如果一个用户每个有五十个,那么它将是125,000 - 很多额外的处理。

答案 1 :(得分:4)

    SELECT u.user_id, 
           u.user_name,
           COUNT(DISTINCT p.post_id) AS `postCount`,
           COUNT(DISTINCT ph.photo_id) AS `photoCount`,
           COUNT(DISTINCT v.video_id) AS `videoCount`
      FROM user u
 LEFT JOIN post p
        ON p.user_id = u.user_id
 LEFT JOIN photo ph
        ON ph.user_id = u.user_id
 LEFT JOIN video v
        ON v.user_id = u.user_id
  GROUP BY u.user_id
  ORDER BY postCount;

Live DEMO

答案 2 :(得分:1)

你的答案可能很慢,因为它正在使用相关的子查询,即子查询为每个user_id运行一次(除非优化器正在做一些聪明的事情 - 不应该依赖它)。

您可以使用左外连接并计算或使用临时内容,例如:

    SELECT u.user_id, 
           u.user_name, 
           ph.user_count AS 'photoCount', 
           p.user_count AS 'postCount', 
           v.user_count AS 'videoCount'
      FROM user u
INNER JOIN (  SELECT user_id,
                     COUNT(*) AS user_count 
                FROM photo 
            GROUP BY user_id
           ) ph
        ON ph.user_id=u.user_id
INNER JOIN (  SELECT user_id, 
                     COUNT(*) AS user_count 
                FROM post
            GROUP BY user_id
           ) p 
        ON p.user_id=u.user_id
INNER JOIN (  SELECT user_id, 
                     COUNT(*) AS user_count 
                FROM video 
            GROUP BY user_id
           ) v
        ON v.user_id=u.user_id

两者都有利弊(取决于索引)。始终查看查询计划(使用EXPLAIN for MySQL)。