数据库SQL实战(题源:牛客网)
数据库SQL实战
1.查找最晚入职员工的所有信息
题目描述
查找最晚入职员工的所有信息
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
));
实现
这种方法使用子查询,适用于可能有多个人都是同一天最晚入职的情况
select *
from employees
where hire_date=
(select max(hire_date)
from employees)
如果最晚一天只有一个人入职,也可以使用 排序+limit
LIMIT m,n : 表示从第m+1条开始,取n条数据;
LIMIT n : 表示从第0条开始,取n条数据,是limit(0,n)的缩写
select *
from employees
order by hire_date desc
limit 1
2.查找入职员工时间排名倒数第三的员工所有信息
题目描述
查找入职员工时间排名倒数第三的员工所有信息
表同1,仍然是emplyees
实现
默认倒数第三入职的员工只有1个人吧
注意日期可能重复,所以需要用"distinct"去重,然后 排序+limit
select *
from employees
where hire_date=
(select distinct hire_date
from employees
order by hire_date desc
limit 2,1
)
3.查找当前薪水详情以及部门编号dept_no
题目描述
查找各个部门当前(to_date=‘9999-01-01’)领导当前薪水详情以及其对应部门编号dept_no
CREATE TABLE dept_manager
(dept_no
char(4) NOT NULL,emp_no
int(11) NOT NULL,from_date
date NOT NULL,to_date
date NOT NULL,
PRIMARY KEY (emp_no
,dept_no
));
CREATE TABLE salaries
(emp_no
int(11) NOT NULL,salary
int(11) NOT NULL,from_date
date NOT NULL,to_date
date NOT NULL,
PRIMARY KEY (emp_no
,from_date
));
输出描述:(输出顺序)
emp_no salary from_date to_date dept_no
实现
select语句后面的一定要和给出的输出顺序一致
其次,from后面的表也应该先写salaries表(我不是很理解为什么这个顺序会有影响)
where条件语句里:两个表中都要to_date属性,所以两个表都要满足条件
select s.*,d.dept_no
from salaries s,dept_manager d
where s.to_date='9999-01-01'
and d.to_date='9999-01-01'
and s.emp_no=d.emp_no
4.查找所有已经分配部门的员工的last_name和first_name
题目描述
查找所有已经分配部门的员工的last_name和first_name
CREATE TABLE dept_emp
(emp_no
int(11) NOT NULL,dept_no
char(4) NOT NULL,from_date
date NOT NULL,to_date
date NOT NULL,
PRIMARY KEY (emp_no
,dept_no
));
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
));
输出描述:
last_name first_name dept_no
实现
考察两个表的连接,
- INNER JOIN自然连接/内连接: 两边表同时有对应的数据,即任何一边缺失数据就不显示。
- LEFT JOIN 左外连接:会读取左边数据表的全部数据,即便右边表无对应数据。
- RIGHT JOIN 右外连接:会读取右边数据表的全部数据,即便左边表无对应数据。
本题要求仅显示符合条件的记录,所以是自然连接
select e.last_name,e.first_name,d.dept_no
from employees e inner join dept_emp d on e.emp_no=d.emp_no
5.查找所有员工的last_name和first_name以及对应部门编号dept_no
题目描述
查找所有员工的last_name和first_name以及对应部门编号dept_no,也包括展示没有分配具体部门的员工
表同4
实现
本题要求除了符合条件的记录,还要显示employees表的全部记录,所以让employees左外连接dept_emp
select e.last_name,e.first_name,d.dept_no
from employees e left outer join dept_emp d on e.emp_no=d.emp_no
6.查找所有员工入职时候的薪水情况
题目描述
查找所有员工入职时候的薪水情况,给出emp_no以及salary, 并按照emp_no进行逆序
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
));
CREATE TABLE salaries
(emp_no
int(11) NOT NULL,salary
int(11) NOT NULL,from_date
date NOT NULL,to_date
date NOT NULL,
PRIMARY KEY (emp_no
,from_date
));
输出描述:
emp_no salary
实现
隐藏条件:employees表的hire_date应该等于salaries表的from_date
select e.emp_no,s.salary
from employees e,salaries s
where e.emp_no=s.emp_no and e.hire_date=s.from_date
order by e.emp_no desc
7.查找薪水涨幅超过15次的员工号emp_no以及其对应的涨幅次数t
题目描述
查找薪水涨幅超过15次的员工号emp_no以及其对应的涨幅次数t
CREATE TABLE salaries
(emp_no
int(11) NOT NULL,salary
int(11) NOT NULL,from_date
date NOT NULL,to_date
date NOT NULL,
PRIMARY KEY (emp_no
,from_date
));
输出描述:
emp_no t
实现
如何体现薪水有涨幅(按题意涨幅可能有正有负)?只要员工工号出现一次,就代表薪水变化了一次
所以统计涨幅次数——就是统计某个员工工号出现的次数
select emp_no,count(emp_no) t
from salaries
group by emp_no
having t>15
8.找出所有员工当前薪水salary情况
题目描述
找出所有员工当前(to_date=‘9999-01-01’)具体的薪水salary情况,对于相同的薪水只显示一次,并按照逆序显示
表同6
实现
注意去重
select distinct salary
from salaries
where to_date='9999-01-01'
order by salary desc
9.获取所有部门当前manager的当前薪水情况,给出dept_no, emp_no以及salary,当前表
题目描述
找出所有员工当前(to_date=‘9999-01-01’)具体的薪水salary情况,对于相同的薪水只显示一次,并按照逆序显示
获取所有部门当前manager的当前薪水情况,给出dept_no, emp_no以及salary,当前表示to_date=‘9999-01-01’
CREATE TABLE dept_manager
(dept_no
char(4) NOT NULL,emp_no
int(11) NOT NULL,from_date
date NOT NULL,to_date
date NOT NULL,
PRIMARY KEY (emp_no
,dept_no
));
CREATE TABLE salaries
(emp_no
int(11) NOT NULL,salary
int(11) NOT NULL,from_date
date NOT NULL,to_date
date NOT NULL,
PRIMARY KEY (emp_no
,from_date
));
输出描述:
dept_no emp_no salary
实现
两个表都有to_date,所以都要满足条件
select d.dept_no,d.emp_no,s.salary
from dept_manager d,salaries s
where d.emp_no=s.emp_no and d.to_date='9999-01-01' and s.to_date='9999-01-01'
10.获取所有非manager的员工emp_no
题目描述
获取所有非manager的员工emp_no
CREATE TABLE dept_manager
(dept_no
char(4) NOT NULL,emp_no
int(11) NOT NULL,from_date
date NOT NULL,to_date
date NOT NULL,
PRIMARY KEY (emp_no
,dept_no
));
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
));
实现
使用not in
select emp_no
from employees
where emp_no not in
(select emp_no
from dept_manager
)
使用in效率较低,也可以使用左连接
将两表连接起来,再找到连接结果中没有部门号(即,非管理者)的员工工号
select e.emp_no
from employees e left join dept_manager d
on e.emp_no=d.emp_no
where d.dept_no is null
上一篇: 《牛客网在线编程数据库SQL实战》