「这是我参与11月更文挑战的第2天,活动详情查看:2021最后一次更文挑战」
描述
有一个员工employees表简况如下:
请你查找employees里入职员工时间排名倒数第三的员工所有信息,以上例子输出如下:
示例1
drop table if exists `employees` ;
CREATE TABLE `employees` (
`emp_no` int(11) NOT NULL,
`birth_date` date NOT NULL,
`first_name` varchar(14) NOT NULL,
`last_name` varchar(16) NOT NULL,
`gender` char(1) NOT NULL,
`hire_date` date NOT NULL,
PRIMARY KEY (`emp_no`));
INSERT INTO employees VALUES(10001,'1953-09-02','Georgi','Facello','M','1986-06-26');
INSERT INTO employees VALUES(10002,'1964-06-02','Bezalel','Simmel','F','1985-11-21');
INSERT INTO employees VALUES(10003,'1959-12-03','Parto','Bamford','M','1986-08-28');
INSERT INTO employees VALUES(10004,'1954-05-01','Chirstian','Koblick','M','1986-12-01');
INSERT INTO employees VALUES(10005,'1955-01-21','Kyoichi','Maliniak','M','1989-09-12');
INSERT INTO employees VALUES(10006,'1953-04-20','Anneke','Preusig','F','1989-06-02');
INSERT INTO employees VALUES(10007,'1957-05-23','Tzvetan','Zielinski','F','1989-02-10');
INSERT INTO employees VALUES(10008,'1958-02-19','Saniya','Kalloufi','M','1994-09-15');
INSERT INTO employees VALUES(10009,'1952-04-19','Sumant','Peac','F','1985-02-18');
INSERT INTO employees VALUES(10010,'1963-06-01','Duangkaew','Piveteau','F','1989-08-24');
INSERT INTO employees VALUES(10011,'1953-11-07','Mary','Sluis','F','1990-01-22');
输出:10005|1955-01-21|Kyoichi|Maliniak|M|1989-09-12
方法一
使用雇佣日期排序加偏移量
SELECT * FROM employees ORDER BY hire_date DESC LIMIT 2, 1
方法二
如果同一天入职的员工不止一个人,那么方法一就不能全部查询出来;
SELECT *
FROM employees
WHERE
hire_date=(SELECT DISTINCT hire_date FROM employees ORDER BY hire_date DESC limit 2, 1)
利用方法一查询出来的日期,再去查一遍表;
方法三
排序问题的利器:窗口函数
SELECT emp_no, birth_date, first_name, last_name, gender, hire_date
FROM
(SELECT *, DENSE_RANK() OVER(ORDER BY hire_date DESC) AS n FROM employees) a
WHERE n=3
使用DENSE_RANK()函数,排序字段如果有重复元素,那么是一条记录占一个序号;
运行效果
方法二:
打开optimizer_trace,查看一下执行流程:select_id为2的子查询会在内存的堆中创建一个临时表,使用优先队列进行排序;(一共有三种排序方法,分别是基于内存的快速排序和优先队列,基于磁盘的归并排序,此处加了limit且数据量少所以用的是优先队列)