博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
leetcode-184-Department Highest Salary 优化记录
阅读量:5252 次
发布时间:2019-06-14

本文共 4233 字,大约阅读时间需要 14 分钟。

题目

The Employee table holds all employees. Every employee has an Id, a salary, and there is also a column for the department Id.

+----+-------+--------+--------------+| Id | Name  | Salary | DepartmentId |+----+-------+--------+--------------+| 1  | Joe   | 70000  | 1            || 2  | Henry | 80000  | 2            || 3  | Sam   | 60000  | 2            || 4  | Max   | 90000  | 1            |+----+-------+--------+--------------+

The Department table holds all departments of the company.

+----+----------+| Id | Name     |+----+----------+| 1  | IT       || 2  | Sales    |+----+----------+

Write a SQL query to find employees who have the highest salary in each of the departments. For the above tables, Max has the highest salary in the IT department and Henry has the highest salary in the Sales department.

+------------+----------+--------+| Department | Employee | Salary |+------------+----------+--------+| IT         | Max      | 90000  || Sales      | Henry    | 80000  |+------------+----------+--------+
先后写了5,6个版本,效率各不相同,挑出典型的5个,来分析一下sql语句的优化

1.Runtime: 1539 ms

select Department.Name as Department,     Employee.Name as Employee,     Employee.Salary as Salary from Department join Employee    on Department.Id = Employee.DepartmentId where (Department.Id, Employee.Salary) in    (select DepartmentId, max(Salary) from Employee group by DepartmentId);

2.Runtime: 1204 ms

select Department.Name as Department,     Employee.Name as Employee,     Employee.Salary as Salary from Department join Employee    on Department.Id = Employee.DepartmentId where (Department.Id, Employee.Salary) in    (select DepartmentId, Salary     from (select * from Employee order by Salary desc) q     group by DepartmentId);

3.Runtime: 1399 ms

select a.Name as Department,     b.Name as Employee,     b.Salary as Salary from Department a join Employee b    on a.Id = b.DepartmentId where exists(select 1 from (select * from Employee order by Salary desc) c         group by DepartmentId         having a.Id = c.DepartmentId and b.Salary = max(c.Salary));

4.Runtime: 980 ms

select a.Name as Department,     b.Name as Employee,     b.Salary as Salary from (Department a join Employee b on a.Id = b.DepartmentId) join    (select c.DepartmentId,max(c.Salary) as Salary from (select * from Employee order by Salary desc) c group by DepartmentId) d    on a.Id = d.DepartmentId and b.Salary = d.Salary;

5.Runtime: 957 ms

select a.Name as Department,     b.Name as Employee,     b.Salary as Salary from (Department a straight_join Employee b on a.Id = b.DepartmentId) straight_join     (select c.DepartmentId,max(c.Salary) as Salary from (select * from Employee order by Salary desc) c group by c.DepartmentId) d     on a.Id = d.DepartmentId and b.Salary = d.Salary;

 总结

  • 1与2比较,聚合函数 max() 的效率不如嵌套子查询
  • 2与3比较, in 与 exists 效率差不多,当时在网上查的是:

 1、in 和 not in 也要慎用,否则会导致全表扫描

 2、很多时候用 exists 代替 in 是一个好的选择

  不过通过后面的优化,可以看出 in 确实挺慢的

  • 3与4比较,4用 join on 代替了 where 判断,效率提升很多,后来有个看过mysql源码的大神说:

 在 MySQL 的 SELECT 查询当中,其核心算法就是 JOIN 查询算法。其他的查询语句都相应向 JOIN 靠拢:单表查询被当作 JOIN 的特例;子查询被尽量转换为 JOIN 查询

  • 4与5比较,5将 join 替换为了 straight_join ,还是源码大神说的:

对于多表查询,如果可以确定表按照某一固定次序处理可以获得较好的效率,则建议加上 STRAIGHT_JOIN 子句,以减少优化器对表进行重排序优化的过程。

该子句一方面可以用于优化器无法给出最优排列的 SQL 语句;另一方面同样适用于优化器可以给出最优排列的 SQL 语句,因为 MySQL 算出最优排列也需要耗费较长的流程。

对于后一状况,可以根据 EXPLAIN 的提示选定表的顺序,并加上 STRAIGHT_JOIN 子句固定该顺序。该状况下的使用前提是几个表之间的数据量比例会一直保持在某一顺序,否则在各表数据此消彼长之后会适得其反。

  对于经常调用的 SQL 语句,这一方法效果较好;同时操作的表越多,效果越好。

后记

  至此,优化还没完全结束,leetcode上该题最快是813ms,但是没有分享代码,最后贴两个别人家的代码:

  Join twice,890ms accepted

SELECT Name, Employee, SalaryFROM Department JOIN (SELECT Employee.Name AS Employee, Employee.Salary, Employee.DepartmentId    FROM Employee JOIN (SELECT `DepartmentId`, MAX(`Salary`) AS Salary        FROM `Employee`        GROUP BY `DepartmentId`        ) t1 ON t1.DepartmentId = Employee.DepartmentId    AND t1.Salary = Employee.Salary    ) t2 ON Department.Id = t2.DepartmentId

  Easy Solution. No joins. GROUP BY is enough. 916ms

selectd.Name, e.Name, e.SalaryfromDepartment d,Employee e,(select MAX(Salary) as Salary,  DepartmentId as DepartmentId from Employee GROUP BY DepartmentId) hwheree.Salary = h.Salary ande.DepartmentId = h.DepartmentId ande.DepartmentId = d.Id;

转载于:https://www.cnblogs.com/zhangyunhao/p/4896055.html

你可能感兴趣的文章
织梦仿站第三课:网站的文件分割
查看>>
Windows 2003全面优化
查看>>
URAL 1002 Phone Numbers(KMP+最短路orDP)
查看>>
web_day4_css_宽度
查看>>
用sql删除数据库重复的数据的方法
查看>>
学习笔记21—PS换图片背景
查看>>
electron入门心得
查看>>
格而知之2:UIView的autoresizingMask属性探究
查看>>
Spring3.0 AOP 具体解释
查看>>
我的Hook学习笔记
查看>>
EasyUI DataGrid 中字段 formatter 格式化不起作用
查看>>
海量数据存储
查看>>
js中的try/catch
查看>>
[导入]玫瑰丝巾!
查看>>
自动从网站上面下载文件 .NET把网站图片保存到本地
查看>>
【识记】 域名备案
查看>>
STL uva 11991
查看>>
MY SQL的下载和安装
查看>>
自定义OffMeshLink跳跃曲线
查看>>
寄Android开发Gradle你需要知道的知识
查看>>