[精讚] [會員登入]
624

【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

你可能感興趣的文章

【JDA/discord bot】如何獲得訊息中的圖片和影片並儲存或轉傳 如何將訊息中的圖片或影片之類的多媒體提取出來之後,再進行相關的處理

【CoreProtect】自行升級至 Minecraft 1.21.x、1.22.x 或更高版本 不知道為什麼,官方竟然無預警停止更新,我們只好自行救濟

【Maven】如何夾帶檔案至Jar內以及其讀取方式 想要讀取一個外部的文字文件?Maven是你的好幫手

【D語言】D language 安裝及使用 一種比較少見的程式語言

【C++】class練習 — 檢測該字串是否為迴圈 第一次的C++結構式寫法,雖然以後應該會見怪不怪,但畢竟是第一次所以還是想保存下來

【Nexus Repository Manager】(deploy)使他人可以對遠端資料庫做讀寫 使用 Nexus Repository Manager 來讓各個工程師控制自己的 Jar 包,不會有 Github Merge Crashed 問題。

隨機好文

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

高捷少女:小穹與果仁巧克力㊤ 阿敏突然輕笑一聲,從櫃臺拿來一個塑膠餐盒,打開給大家看。「這是小穹烤的餅乾,妳們吃吃看就知道她為什麼不想講了。」小穹變得緊張起來。「阿敏,妳怎麼還留著呀?」艾米莉亞、婕兒與耐耐各自拿了一塊,把夾著奶油

高捷少女:美麗島的守護者① 婕兒跟耐耐嚇得臉色發白。「耐耐,小穹有這方面的愛好喔?」婕兒小聲地說。耐耐默默的拿出筆記本,臉上浮現一層微笑:「這麼大的新聞,要趕快記下來!」

高捷少女:美麗島的守護者③ 小雅閉上眼睛,思索在高捷發生的點點滴滴。她心意已決,在高捷的日子的確也有快樂的部分,不過她相信換個方向是更好的決定。有關高捷的所有美好回憶,小雅決定保留在心裡就好,繼續在高捷工作只會讓自己更痛苦而已,

高捷少女:美麗島的守護者(終) 「我是她的師娘,妳一定就是艾米莉亞吧?小雅昨晚有跟我說到您。」「師娘?」艾米疑惑不解。