MySQL查询每个部门薪水TOP3

本文介绍了一种复杂的SQL查询方法,用于从Employee表中找出每个部门薪酬最高的前三名员工,并展示了如何通过多层子查询实现这一目标。

假设有两个表Department和Employee。

department的表结构如下:

+-------+-------------+------+-----+---------+----------------+
| Field | Type        | Null | Key | Default | Extra          |
+-------+-------------+------+-----+---------+----------------+
| Id    | int(11)     | NO   | PRI | NULL    | auto_increment |
| Name  | varchar(60) | YES  |     | NULL    |                |
+-------+-------------+------+-----+---------+----------------+

employee表结构如下:

+--------------+-------------+------+-----+---------+----------------+
| Field        | Type        | Null | Key | Default | Extra          |
+--------------+-------------+------+-----+---------+----------------+
| Id           | int(11)     | NO   | PRI | NULL    | auto_increment |
| Name         | varchar(60) | YES  |     | NULL    |                |
| Salary       | double(9,2) | YES  |     | NULL    |                |
| DepartmentId | int(11)     | YES  | MUL | NULL    |                |

+--------------+-------------+------+-----+---------+----------------+


现在要查询每个部门的薪水TOP3,查询语句如下:

SELECT
	d. NAME Department,
	c. NAME Employee,
	c.Salary
FROM
	(
		SELECT
			b.DepartmentId,
			b.Salary,
			b. NAME,
			b.rank
		FROM
			(
				SELECT
					a.DepartmentId,
					a.Salary,
					a. NAME,

				IF (
					@pa = a.DepartmentId,

				IF (
					@pb = a.Salary ,@rank :=@rank ,@rank :=@rank + 1
				) ,@rank := 1
				) rank ,@pa := a.DepartmentId ,@pb := a.Salary
				FROM
					(
						SELECT
							@rank := 0,
							DepartmentId,
							Salary,
							NAME
						FROM
							Employee
						ORDER BY
							DepartmentId,
							Salary DESC
					) a
			) b
		HAVING
			rank <= 3
	) c
JOIN Department d ON (c.DepartmentId = d.Id)
ORDER BY
	c.DepartmentId,
	c.Salary DESC


### MySQL 数据库查询操作示例 #### 基本查询 为了获取表中的所有记录,可以使用简单的 `SELECT` 语句。例如,假设有一个名为 `employees` 的表,则可以通过下面的 SQL 语句来检索该表内的所有数据: ```sql SELECT * FROM employees; ``` 当只需要特定列的信息时,可以在 `SELECT` 后面指定这些列的名字[^1]。 #### 使用 DISTINCT 关键词消除重复项 如果希望返回的结果集中不含有任何冗余的数据行,可利用 `DISTINCT` 来实现这一点。比如想要得到公司内部独一无二的工作职位列表,应该这样写: ```sql SELECT DISTINCT job_title FROM employees; ``` 这会确保每一个工作头衔只会出现一次[^2]。 #### 添加 WHERE 子句进行条件筛选 通过附加 `WHERE` 子句给查询提供额外的选择标准,从而精确控制所选出来的记录范围。考虑这样一个场景:找出工资超过80,000美元的所有员工姓名及其薪水数额: ```sql SELECT first_name, last_name, salary FROM employees WHERE salary > 80000; ``` 此命令仅显示满足设定阈值以上的雇员资料。 #### 排序结果集 ORDER BY 为了让最终输出按照某种顺序排列,可在查询结尾处追加 `ORDER BY` 子句。继续上面的例子,现在还希望能够按降序展示上述人员名单: ```sql SELECT first_name, last_name, salary FROM employees WHERE salary > 80000 ORDER BY salary DESC; ``` 这里的 `DESC` 表明是从高到低排序;如果不加上参数,默认情况下将是升序 (`ASC`) . #### 聚合函数的应用 对于统计分析来说,聚合函数是非常有用的工具之一。它们允许计算诸如总和、平均数之类的数值型属性。举例而言,想知道整个部门成员薪资总额的话,可以用如下方式表达需求: ```sql SELECT SUM(salary) AS total_salary FROM employees; ``` 这段代码不仅执行了求和动作,而且借助别名机制让输出更易于理解. #### JOIN 操作关联多张表格 有时候单靠一张表无法完成复杂的业务逻辑处理,这时就需要把几张有关联性的表结合起来看。以订单管理和客户信息为例,假设有两张分别存储着顾客详情(`customers`)以及他们下的定单(`orders`)的表,那么要找到某位具体客户的最近一笔交易日期就可以这样做: ```sql SELECT c.customer_id, o.order_date FROM customers c INNER JOIN orders o ON c.customer_id = o.customer_id WHERE c.first_name=&#39;John&#39; AND c.last_name=&#39;Smith&#39;; ``` 这里采用了内连接(inner join),意味着只有那些能在双方都找到匹配记录的情况才会被保留下来参与后续的操作. #### 子查询嵌套 某些复杂的需求可能涉及到先从某个集合里面挑出一部分符合条件的对象作为下一步骤的基础素材。这时候就轮到了子查询出场发挥作用了。设想一下这样的情况:寻找在过去一年里购买次数最多的产品类别。为此,首先要识别出活跃度最高的几位买家群体,然后再基于这些人再去考察其消费偏好分布状况: ```sql WITH top_buyers AS ( SELECT customer_id FROM purchases GROUP BY customer_id HAVING COUNT(*) >= ALL (SELECT COUNT(*) FROM purchases GROUP BY customer_id) ) SELECT p.category FROM products p JOIN purchases pu ON p.product_id=pu.product_id JOIN top_buyers tb ON pu.customer_id=tb.customer_id GROUP BY p.category ORDER BY COUNT(pu.purchase_id) DESC LIMIT 1; ``` 以上例子展示了如何构建一个带有窗口功能(CTE)并结合多个层次化的子查询结构来进行深入挖掘的过程.
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值