[精讚] [會員登入]
614

【MySQL】每個類別中取前三高

一個由多層次所組成的SELECT述句

分享此文連結 //n.sfs.tw/16155

分享連結 【MySQL】每個類別中取前三高@小編過路君子
(文章歡迎轉載,務必尊重版權註明連結來源)
2023-04-11 23:22:54 最後編修
2023-04-10 15:06:36 By 過路君子
 

大家好,這裡是快忘光SQL的小編過路君子

看了看紀錄,上次寫這麼複雜的SQL已經是在一年之前

 

 

SQL 小編說常用也常用,說不常用也不常用,為什麼這麼說呢?

因為平常使用的時候就只會用基本的 SELECT 功能,什麼 sub select 根本就不會用到。

今天回來複習一下 MySQL 的 SQL 語法

 

那在開始之前,先假設我們有以下的資料表:

Table: Employee
+--------------+---------+
| Column Name  | Type    |
+--------------+---------+
| id           | int     |
| name         | varchar |
| salary       | int     |
| departmentId | int     |
+--------------+---------+

Table: Department
+-------------+---------+
| Column Name | Type    |
+-------------+---------+
| id          | int     |
| name        | varchar |
+-------------+---------+

並且具有以下的內容:

Employee table:
+----+-------+--------+--------------+
| id | name  | salary | departmentId |
+----+-------+--------+--------------+
| 1  | Joe   | 85000  | 1            |
| 2  | Henry | 80000  | 2            |
| 3  | Sam   | 60000  | 2            |
| 4  | Max   | 90000  | 1            |
| 5  | Janet | 69000  | 1            |
| 6  | Randy | 85000  | 1            |
| 7  | Will  | 70000  | 1            |
+----+-------+--------+--------------+
Department table:
+----+-------+
| id | name  |
+----+-------+
| 1  | IT    |
| 2  | Sales |
+----+-------+

 

不考慮資料重複或並列

我們則需要輸出以下的資料:

+------------+----------+--------+
| Department | Employee | Salary |
+------------+----------+--------+
| IT         | Max      | 90000  |
| IT         | Joe      | 85000  |
| IT         | Randy    | 85000  |
| Sales      | Henry    | 80000  |
| Sales      | Sam      | 60000  |
+------------+----------+--------+

簡單來說就是將 departmentId 轉換成對應的部門名稱,然後要將每個前三高薪水(Salary)的人列出,最後在對資料做排序。

 

直接上 SQL 語法:

SELECT c.name AS Department, a.name AS Employee, a.salary AS Salary
FROM Employee AS a
LEFT JOIN Department AS c
  ON c.id = a.departmentId
WHERE
  (
    SELECT COUNT(b.salary)
    FROM Employee AS b
    WHERE b.departmentId = a.departmentId
      AND a.salary < b.salary
  ) < 3
ORDER BY c.id, a.salary;

輸出結果就會如上方所示。

其中的 LEFT JOIN 作用就是為了要讓 Department 表裡的資料直接接在 Employee 表的後面,只有當 Department 表內的 id 和 Employee 表內的 departmentId 相同才需要做串接。

而比較重要的就是 WHERE 內的 sub select 了,此 sub select 會試著去統計在 Employee 表內目前有幾個人的薪水大於目前所執行到人的薪水。

而最後的小於三則是取前三高,如果要取前五高就改成 5 即可。

 

考慮資料重複或並列

若需考慮資料重複,那我們要輸出的資料變成如下:

+------------+----------+--------+
| Department | Employee | Salary |
+------------+----------+--------+
| IT         | Max      | 90000  |
| IT         | Joe      | 85000  |
| IT         | Randy    | 85000  |
| IT         | Will     | 70000  |
| Sales      | Henry    | 80000  |
| Sales      | Sam      | 60000  |
+------------+----------+--------+

這應該才算是比較常用到的手段,總之寫法如下:

SELECT c.name AS Department, a.name AS Employee, a.salary AS Salary
FROM Employee AS a
LEFT JOIN Department AS c
  ON c.id = a.departmentId
WHERE
  (
    SELECT COUNT(b.salary)
    FROM (SELECT DISTINCT d.departmentId, d.salary FROM Employee AS d) AS b
    WHERE b.departmentId = a.departmentId
      AND a.salary < b.salary
  ) < 3;
ORDER BY c.id, a.salary;

 

只需要改一行就可以了。

簡單來說,在開始找有幾個人薪水大於目前的人之前,先把要比較的表去除重複的資料,這樣統計的時候也不會重複統計啦。

 

 

 

後記

總之這次寫完感覺這個 SQL 很有用,所以就在這邊留存一份,省的未來在那邊推敲半天。

END

你可能感興趣的文章

【Discord bot】[botton]按鈕的使用、響應和關閉 Discord的botton通常都要和View配合使用。

【ThinkSpeak】一個IoT數據分析(IoT analytics)及呈現的網站 世界上有著各式各樣的需求,因而誕生出了各式各樣的網站,絕對不是沒有人提供,而是你不知道哪裡有這東西

【JAVA】[java.lang.Class]如何不透過new來實體化class 在某些極端狀態下,我們沒有辦法直接將實體化後的 Object 傳入函數內,只能採用傳入 Class 在實體化成物件這種極端的方式來進行

【ESXI & Oepnwrt】基本錯誤處理 Oepnwrt 為一個自由度超高的軟體,很多事情都能請其幫忙完成,但也因此很複雜,這邊簡單的介紹幾個錯誤的處理方式

【Maven】如何創建一個簡單可執行的JAR檔 滿重要的一個小功能,畢竟有時候是要傳遞的是JAR檔,而不是直接透過Maven直接部屬之類的

【Java】[debug]多維泛型的大坑 incompatible types error

隨機好文

【手遊介紹】小品手遊─寶箱是我的!(SUMMONER'S GREED) 輕鬆無腦的塔防遊戲,殺時間本小編推薦的遊戲之一

希萌創意預計在今年7月繼東津萌米之後再次推出新遊戲--食用性少女! 今天來介紹希萌創意的心企劃案,來讓大家知道這個消息!讓大家的錢包君一起來減肥吧!Ψ(☆w☆)

高捷少女:布拉格體驗㊦ 「各位想到盧卡站的乘客,請到我們左手邊排隊!」婕兒大聲地喊道。     「這孩子怎麼穿著地鐵站制服?童工嗎?」一位大嬸歪頭問道。

【歌評】蓮台野夜行 - 幻視之夜 ~ Ghostly Eyes 若有什麼是在聽到boss曲前的鋪襯,那一定就是每個系列的道中曲

【歌評】蓮台野夜行 - 少女秘封倶楽部 少女秘封俱樂部 我們的主角終於出場啦!這首歌一反蓮台野夜行的固有風格,總是華麗的開始又華麗的結束!