-
Notifications
You must be signed in to change notification settings - Fork 118
/
Copy path1194_Tournament_Winners.sql
80 lines (71 loc) · 2.34 KB
/
1194_Tournament_Winners.sql
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
-- Solution 1: Subquery, Join, Window Function
-- merge first and second players and their scores into two columns
WITH tb1 AS (
SELECT first_player AS player, first_score as score
FROM Matches
UNION ALL
SELECT second_player, second_score
FROM Matches
),
-- calculate total points scored by each user along with their group information
tb2 AS (
SELECT p.player_id, p.group_id, SUM(tb1.score) AS tp
FROM Players p
LEFT JOIN tb1
ON p.player_id = tb1.player
GROUP BY p.player_id, p.group_id
)
-- find the winner with lowest player_id in each group who scored the maximum total points within the group
SELECT group_id, player_id
FROM (
SELECT player_id, group_id,
ROW_NUMBER() OVER (PARTITION BY group_id ORDER BY tp DESC, player_id) AS r
FROM tb2
) tb3
WHERE r = 1;
-- Solution 2: Subquery, Join, CASE WEHN, Window Function
-- calculate total points scored by each user along with their group information
WITH tb1 AS (
SELECT player_id, group_id,
SUM(
CASE
WHEN player_id = first_player THEN first_score
ELSE second_score
END
) AS tp
FROM players p
LEFT JOIN matches m
ON p.player_id = m.first_player OR p.player_id = m.second_player
GROUP BY player_id, group_id
)
-- find the winner with lowest player_id in each group who scored the maximum total points within the group
SELECT group_id, player_id
FROM (
SELECT player_id, group_id,
ROW_NUMBER() OVER (PARTITION BY group_id ORDER BY tp DESC, player_id) AS r
FROM tb1
) tb2
WHERE r = 1;
-- Solution 3: Subquery, Join, CASE WEHN
-- calculate total points scored by each user along with their group information
WITH tb1 AS (
SELECT player_id, group_id,
SUM(
CASE
WHEN player_id = first_player THEN first_score
ELSE second_score
END
) AS tp
FROM players p
LEFT JOIN matches m
ON p.player_id = m.first_player OR p.player_id = m.second_player
GROUP BY player_id, group_id
)
-- find the winner with lowest player_id in each group who scored the maximum total points within the group
SELECT l.group_id, l.player_id
FROM tb1 l
JOIN tb1 r
ON l.group_id = r.group_id AND
(l.tp < r.tp OR (l.tp = r.tp AND l.player_id >= r.player_id))
GROUP BY l.group_id, l.player_id
HAVING COUNT(*) = 1;