【问题标题】:Sql grouping querysql分组查询
【发布时间】:2016-09-20 07:02:30
【问题描述】:

我遇到了一个 SQL 查询。我知道要做什么,但我不知道怎么做。 所以,这里是需要的方案:

电影(标题导演、年份、类型、费率);

TITLE         DIRECTOR     YEAR      GENRE      RATE 
Fight club    Fincher      1999      Action     4.5
Vertigo       Hitchock     1958      Drama      5
Donnie darko  Kelly        2001      Thriller   3.5   

视频(title, 导演, colloc);

TITLE          DIRECTOR    COLLOC
Fight club     Fincher     3877
Fight club     Fincher     3878
Vertigo        Hitchcock   5431
Vertigo        Hitchcock   5432
Donnie darko   Kelly       9986

Rent(colloc, dateRent, customer, dateReturn);

  COLLOC      DATERENT     CUSTOMER     DATERETURN 
  3877        2016-05-02   324          2016-05-04
  3877        2016-05-20   365          2016-05-20
  3878        2016-04-11   876          2016-04-12
  3878        2016-06-06   112          2016-06-08
  ...         ...          ...          ...
  ...         ...          ...          ...
  9986        2016-02-24   443          2016-02-28

这里是查询:

列出每部电影至少租借了两次的视频数量。

(注意:商店通常有更多视频 - dvds、vhs 等 - 用于每部电影)。

我的方法如下:我将从这个简单的查询开始

SELECT colloc, title, director, COUNT(colloc) AS rentNumber
FROM Rent
NATURAL JOIN Video
GROUP BY colloc

要显示这样的内容:

COLLOC    TITLE         DIRECTOR    RENTNUMBER
3877      Fight club    Fincher     2
3878      Fight club    Fincher     2
5432      Vertigo       Hitchcock   2
5431      Vertigo       Hitchcock   1
9986      Donnie darko  Kelly       1 

为了达到这个目的:

TITLE          DIRECTOR      VIDEOSNUMBER
Fight club     Fincher       2
Vertigo        Hitchcock     1
Donnie darko   Kelly         0

但我错过了最后一步,我无法正确使用 GROUP BY / HAVING。 即使使用子查询也没有成功。 我一直在尝试使用不同的方法,但结果并不好,因此我们将不胜感激。

谢谢。

编辑:注意,没有必要在查询中使用表 Movie。我把它放在那里只是为了解释一部电影可以有多个视频。 添加了表格示例。主键以粗体显示。

解决方案: 多亏了您的提示,我找到了一个解决方案(老实说,它是部分的,因为它会放出所有没有至少租用 2 次视频的电影)。 无论如何,这里是:

SELECT title, director, COUNT(title) AS videosNumber
FROM (SELECT colloc, title, director, COUNT(colloc) AS rentNumber
      FROM rent NATURAL JOIN video
      GROUP BY colloc
      ) X
WHERE rentNumber > 1
GROUP BY title

【问题讨论】:

  • 我认为你应该 GROUP BY titolo
  • 一般的 GROUP BY 规则说:如果指定了 GROUP BY 子句,则 SELECT 列表中的每个列引用必须要么标识一个分组列,要么是一个集合函数的参数!
  • 添加样本表数据,产生指定的预期结果!
  • 表和键是什么?你想怎么加入他们?
  • 您能解释一下VIDEOSNUMBER 在您的预期输出中的含义吗?

标签: mysql sql


【解决方案1】:

试试这个查询:

SELECT t1.TITLE,
       t1.DIRECTOR,
       COALESCE(t2.VIDEOSNUMBER, 0) AS VIDEOSNUMBER
FROM Movie t1
LEFT JOIN
(
    SELECT v.TITLE, COUNT(DISTINCT r.COLLOC) AS VIDEOSNUMBER
    FROM Rent r
    INNER JOIN Video v
        ON r.COLLOC = v.COLLOC
    GROUP BY v.TITLE
) t2
   ON t1.TITLE = t2.TITLE
ORDER BY COALESCE(t2.VIDEOSNUMBER, 0) DESC

【讨论】:

    【解决方案2】:

    试试这个

    SELECT 
    -- if you want to add colloc, uncomment below
    -- r.colloc as COLLOC
    m.title as TITLE,
    m.director as DIRECTOR,
    COUNT(r.colloc) as VIDEOSNUMBER
    
    FROM movie m
    LEFT JOIN video v ON m.title = v.title
    LEFT JOIN rent r ON v.colloc = r.colloc
    GROUP BY TITLE
    

    【讨论】:

    • 分组依据无效。将在较新的 MySQL 版本中引发错误。 (并在旧版本中返回任意结果。)
    • 独立于较新 MySQL 版本中的错误,它输出与 Nebi 和 @unnikrishnan-r 相同的结果。
    • @Zeno 你把表名改成你的了吗?你能告诉我们你的错误吗?
    • 我有一个旧版本的 MySQL,对我来说它没有错误,但错误无关紧要。不幸的是,结果是不对的。
    【解决方案3】:

    用他下面的脚本试试。 您应该在GROUP BY 子句中包含SELECT 列表中的列。

    SELECT TITLE,DIRECTOR,MIN(VIDEOSNUMBER) VIDEOSNUMBER
    FROM (
            SELECT v.colloc,v.TITLE,v.DIRECTOR ,COUNT( r.colloc) VIDEOSNUMBER    
            FROM Video v 
            LEFT JOIN Rent r on  v.colloc=r.colloc
            GROUP BY  v.TITLE,v.DIRECTOR,v.colloc)t
    GROUP BY TITLE,DIRECTOR
    

    【讨论】:

    • 它输出与@Nebi 的查询相同的结果。
    • 不是确切的输出,但我认为这是解决问题的正确方法,在 from 子句中“嵌套”结果表。谢谢。
    • 向租表添加左连接,然后重试。查看更新的脚本。
    【解决方案4】:

    测试过:

    SELECT m.TITLE, m.DIRECTOR, COALESCE(rs.Rentnumber, 0) AS VIDEOSNUMBER
    FROM Movie AS m
    LEFT JOIN 
    (
        SELECT 
            --r.colloc, -- if you want to see r.colloc
            v.TITLE, 
            v.DIRECTOR, 
            COUNT(v.TITLE) AS Rentnumber
        FROM Video AS v
        LEFT JOIN Rent as r on v.colloc = r.colloc 
        GROUP BY r.colloc, v.TITLE, v.DIRECTOR
    ) AS rs ON m.title = rs.title AND m.director = rs.director
    Group by m.title, m.director
    

    【讨论】:

    • 这是我尝试过的解决方案之一,但 VIDEOSNUMBER 将包含每部电影的租借总数,而不是租借超过一次的视频数量。另外,不会有没有租借视频的电影。
    • @zeno 已更新查询。没有测试过。它们可能是 MySQL 中更清洁的解决方案。在 SQL-SERVER 中,您为此使用 Window-Functions。在 MySQL 中你可以试试Group_concat
    • 它输出了一些错误,但我认为这可能是一个很好的起点。
    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 2015-09-29
    • 1970-01-01
    • 1970-01-01
    • 2015-08-08
    • 2018-09-02
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多