Skip to content

Latest commit

 

History

History
142 lines (116 loc) · 5.43 KB

File metadata and controls

142 lines (116 loc) · 5.43 KB
comments difficulty edit_url tags
true
困难
数据库

English Version

题目描述

表: Wineries

+-------------+----------+
| Column Name | Type     |
+-------------+----------+
| id          | int      |
| country     | varchar  |
| points      | int      |
| winery      | varchar  |
+-------------+----------+
id 是这张表具有唯一值的列。
这张表包含 id, country, points,和 winery。

编写一个解决方案,根据每家酒庄的 总分 找出 每个国家前三名酒庄。如果有 多个酒庄 的总分 相同,则按 winery 名称升序排列。如果没有 分数排在第二的酒庄,则输出 'No Second Winery',如果没有 分数排在第三的酒庄,则输出 'No Third Winery'。

返回结果表按 country 升序 排列。

结果表格格式如下例所示。

 

示例 1:

输入:
Wineries table:
+-----+-----------+--------+-----------------+
| id  | country   | points | winery          | 
+-----+-----------+--------+-----------------+
| 103 | Australia | 84     | WhisperingPines | 
| 737 | Australia | 85     | GrapesGalore    |    
| 848 | Australia | 100    | HarmonyHill     | 
| 222 | Hungary   | 60     | MoonlitCellars  | 
| 116 | USA       | 47     | RoyalVines      | 
| 124 | USA       | 45     | Eagle'sNest     | 
| 648 | India     | 69     | SunsetVines     | 
| 894 | USA       | 39     | RoyalVines      |  
| 677 | USA       | 9      | PacificCrest    |  
+-----+-----------+--------+-----------------+
输出:
+-----------+---------------------+-------------------+----------------------+
| country   | top_winery          | second_winery     | third_winery         |
+-----------+---------------------+-------------------+----------------------+
| Australia | HarmonyHill (100)   | GrapesGalore (85) | WhisperingPines (84) |
| Hungary   | MoonlitCellars (60) | No second winery  | No third winery      | 
| India     | SunsetVines (69)    | No second winery  | No third winery      |  
| USA       | RoyalVines (86)     | Eagle'sNest (45)  | PacificCrest (9)     | 
+-----------+---------------------+-------------------+----------------------+
解释:
对于 Australia
 - HarmonyHill 酒庄获得了 Australia 的最高分数,为 100 分。
 - GrapesGalore 酒庄总共获得 85 分,位列 Australia 的第二位。
 - WhisperingPines 酒庄总共获得 80 分,位列 Australia 的第三位。
对于 Hungary
 - MoonlitCellars 是唯一的酒庄,获得 60 分,自动成为最高分数的酒庄。没有第二或第三家酒庄。
对于 India
 - SunsetVines 是唯一的酒庄,获得 69 分,成为最高的酒庄。没有第二或第三家酒庄。
对于 USA
 - RoyalVines Wines 累计了总分 47 + 39 = 86 分,占据了 USA 的最高位置。
 - Eagle'sNest 总共获得 45 分,位列 USA 的第二高位置。
 - PacificCrest 累计了 9 分,位列 USA 的第三高酒庄。
输出表按国家首字母升序排列。

解法

方法一:分组 + 窗口函数 + 左连接

我们可以先对 Wineries 表按照 countrywinery 进行分组,计算每个分组的总得分 points,然后再利用窗口函数 RANK() 将数据再按照 country 进行分组,按照 points 降序、winery 升序进行排序,并且用 CONCAT() 函数将 winerypoints 进行拼接,得到如下形式的数据,记为 T 表:

country winery rk
Australia HarmonyHill (100) 1
Australia GrapesGalore (85) 2
Australia WhisperingPines (84) 3
Hungary MoonlitCellars (60) 1
India SunsetVines (69) 1
USA RoyalVines (86) 1
USA Eagle'sNest (45) 2
USA PacificCrest (9) 3

接下来,我们只需要筛选出 rk = 1 的数据,然后再将 T 表自连接两次,分别连接 rk = 2rk = 3 的数据,即可得到最终结果。

MySQL

# Write your MySQL query statement below
WITH
    T AS (
        SELECT
            country,
            CONCAT(winery, ' (', points, ')') AS winery,
            RANK() OVER (
                PARTITION BY country
                ORDER BY points DESC, winery
            ) AS rk
        FROM (SELECT country, SUM(points) AS points, winery FROM Wineries GROUP BY 1, 3) AS t
    )
SELECT
    t1.country,
    t1.winery AS top_winery,
    IFNULL(t2.winery, 'No second winery') AS second_winery,
    IFNULL(t3.winery, 'No third winery') AS third_winery
FROM
    T AS t1
    LEFT JOIN T AS t2 ON t1.country = t2.country AND t1.rk = t2.rk - 1
    LEFT JOIN T AS t3 ON t2.country = t3.country AND t2.rk = t3.rk - 1
WHERE t1.rk = 1
ORDER BY 1;