数据库SQL实战 1-20(题源:牛客网)

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

11.获取所有员工当前的manager

题目描述
获取所有员工当前的manager,如果当前的manager是自己的话结果不显示,当前表示to_date=‘9999-01-01’。
结果第一列给出当前员工的emp_no,第二列给出其manager对应的manager_no。
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 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));
输出描述:
emp_no manager_no
实现
员工和经理,部门号相同。题干说员工号对应的是经理的话就不显示,所以员工和经理的员工号应该不同
综上:员工和经理—— 部门号相同,员工号不同

select de.emp_no emp_no,dm.emp_no manager_no
from dept_emp de,dept_manager dm
where de.dept_no=dm.dept_no and de.emp_no!=dm.emp_no
and de.to_date='9999-01-01' and dm.to_date='9999-01-01'

12.获取所有部门中当前员工薪水最高的相关信息

题目描述
获取所有部门中当前员工薪水最高的相关信息,给出dept_no, emp_no以及其对应的salary
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 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));
实现

select d.dept_no,s.emp_no,max(s.salary) salary
from dept_emp d,salaries s
where d.emp_no=s.emp_no and d.to_date='9999-01-01' and s.to_date='9999-01-01'
group by d.dept_no

13.从titles表获取按照title进行分组

题目描述
从titles表获取按照title进行分组,每组个数大于等于2,给出title以及对应的数目t。
CREATE TABLE IF NOT EXISTS “titles” (
emp_no int(11) NOT NULL,
title varchar(50) NOT NULL,
from_date date NOT NULL,
to_date date DEFAULT NULL);
实现
注意:条件判断使用having

select title,count(emp_no) t
from titles
group by title
having t>=2

14.从titles表获取按照title进行分组,注意对于重复的emp_no进行忽略。

题目描述
从titles表获取按照title进行分组,每组个数大于等于2,给出title以及对应的数目t。
注意对于重复的title进行忽略。
实现
去除重复的员工号

select title,count(distinct emp_no) t
from titles
group by title
having t>=2

15.查找employees表

题目描述
查找employees表所有emp_no为奇数,且last_name不为Mary的员工信息,并按照hire_date逆序排列
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));
输出描述:
emp_no birth_date first_name last_name gender hire_date
实现

select *
from employees 
where emp_no%2!=0 and last_name!='Mary'
order by hire_date desc

16.统计出当前各个title类型对应的员工当前薪水对应的平均工资

题目描述
统计出当前各个title类型对应的员工当前(to_date=‘9999-01-01’)薪水对应的平均工资。结果给出title以及平均工资avg。
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));
CREATE TABLE IF NOT EXISTS “titles” (
emp_no int(11) NOT NULL,
title varchar(50) NOT NULL,
from_date date NOT NULL,
to_date date DEFAULT NULL);
实现

select t.title,avg(s.salary) avg
from titles t,salaries s
where t.emp_no=s.emp_no and t.to_date='9999-01-01' and s.to_date='9999-01-01'
group by t.title

17.获取当前薪水第二多的员工的emp_no以及其对应的薪水salary

题目描述
获取当前(to_date=‘9999-01-01’)薪水第二多的员工的emp_no以及其对应的薪水salary
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));

实现

select emp_no,salary
from salaries
where to_date='9999-01-01'
order by salary desc
limit 1,1

18.获取当前薪水第二多的员工的emp_no以及其对应的薪水salary,不准使用order by

题目描述
查找当前薪水(to_date=‘9999-01-01’)排名第二多的员工编号emp_no、薪水salary、last_name以及first_name,不准使用order by
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));

实现
多层嵌套子查询
最内层子查询找出薪水最高的,然后去掉这个最高的额,再找薪水最高的,就相当于找到了整体薪水第二高的

select e.emp_no,s.salary,e.last_name,e.first_name 
from employees e,salaries s
where e.emp_no=s.emp_no and s.salary=
(select max(salary)
 from salaries
 where salary !=
 (select max(salary)
 from salaries
 )
)

19.查找所有员工的last_name和first_name以及对应的dept_name

题目描述
查找所有员工的last_name和first_name以及对应的dept_name,也包括暂时没有分配部门的员工
CREATE TABLE departments (
dept_no char(4) NOT NULL,
dept_name varchar(40) NOT NULL,
PRIMARY KEY (dept_no));
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));
实现
三表连接
要查询所有员工,所以从employees表开始进行左连接–>dept_emp表–>再左连接departments表

select e.last_name,e.first_name,d.dept_name
from employees e 
left join dept_emp de on e.emp_no=de.emp_no
left join departments d on d.dept_no=de.dept_no

20.查找员工编号emp_no为10001其自入职以来的薪水salary涨幅值growth

题目描述
查找员工编号emp_no为10001其自入职以来的薪水salary涨幅值growth
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));
实现
最大工资-最小工资
(本来想用当前工资-入职工资,但期间工资可能降了,似乎不太符合涨幅的定义)

select max(salary)-min(salary)
from salaries
where emp_no=10001
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值