牛客数据库SQL实战题(1-10题)
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);
这里主要的争论点是,最晚入职的是一个人,还是最晚时间对应的多个人。上述答案是以时间为准,筛选出最晚时间,然后找到对应最晚时间的所有人。
2、查找入职员工时间排名倒数第三的员工所有信息
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 distinct hire_date
from employees
order by hire_date desc
limit 2,1);
这里主要的技巧是使用limit,Limit接受一个或两个数字参数。参数必须是一个整数常量。如果给定两个参数,第一个参数指定第一个返回记录行的偏移量,第二个参数指定返回记录行的最大数目。
select * from table limit 5,10;
select * from table limit 5,-1;
上面的代码第一个是检索第6行到15行,包括第15行,也就是返回从第6行开始的接下来的10行。
第二个是检索从第6行开始直到结束的所有行。-1表示倒数第一个。
3、查找各个部门当前(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));
答案
select sa.*, de.dept_no
from (select * from salaries where to_date='9999-01-01') as sa
join (select * from dept_manager where to_date='9999-01-01') as de
on sa.emp_no=de.emp_no;
这题主要的点是dept_manager和salaries哪个是主表,答案是salaries表是主表,也就是说,当我们join表时,salaries表在左侧。
先在salaries和dept_no表中,选中所有to_date='9999-01-01'的行,也就是过滤掉失效信息,筛选出当前薪水和领导信息。然后题目想要获取的from_date是关于薪水的from_date,而不是dept_manager表的from_date(面向答案编程)。我们要筛选的列是salaries表的所有列,和dept_manager表中的dept_no列。
这题的join
是内连接,写inner join
也可以。outer join
、left join
、right join
不行。
另外,讨论区有人谈到可以用自然连接natural join
。自然连接是一种特殊的等值连接,自然连接在结果中会把重复的属性列去掉。一般的连接操作是从行的角度进行运算,但是自然连接还需要取消重复列,所以是同时从行和列的角度进行运算。
其他答案
select sa.* ,de.dept_no
from salaries as sa
join dept_manager as de
on sa.emp_no=de.emp_no
where sa.to_date ='9999-01-01' and de.to_date='9999-01-01';
4、查找所有已经分配部门的员工的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));
答案
select em.last_name, em.first_name, de.dept_no
from dept_emp as de
join employees as em
on de.emp_no=em.emp_no;
因为dept_emp表中的dept_no非空,dept_emp表保存的一定是已经分配了部门的人。
5、查找所有员工的last_name和first_name以及对应部门编号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));
答案
select em.last_name, em.first_name, de.dept_no
from employees as em
left join dept_emp as de
on em.emp_no=de.emp_no;
这题主要是left join
。
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));
答案
select em.emp_no, sa.salary
from employees as em
join salaries as sa
on em.emp_no=sa.emp_no and em.hire_date=sa.from_date
order by em.emp_no desc;
emplyees表中的主键是emp_no,也就是说一个员工只有一个号码。但是salaries表中的主键是emp_no和from_date,也就是说员工可能变更薪水,所以同一个emp_no可能有多条记录。要查找员工入职时候的薪水,表达式为em.hire_date=sa.from_date
。
上述sql语句写left join也是正确的。
其他答案
select em.emp_no, sa.salary
from employees as em, salaries as sa
where em.emp_no=sa.emp_no and em.hire_date=sa.from_date
order by em.emp_no desc;
这里是直接查询两张表,并没有将表进行join,直接在where中进行筛选。
7、查找薪水涨幅超过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));
答案
select emp_no, count(from_date) as t
from salaries
group by emp_no
having t>15;
这题主要是where和having的区别:
他们的本质的区别是where筛选的是数据库表里面本来就有的字段,而having筛选的字段是从前筛选的字段筛选的。
例如:
select a, b from table where a>5;
select a, b from table having a>5;
//以上两种都正确
select a, b from table where c>3;
select a, b from table having c>3;//错误
//用where正确,但用having错误。
//用where是先筛选行,再select列;但having是先select列,再having筛选行。
//也就是说,如果用having,select的列中a,b没有c,则用having错误。
select a, avg(b) as avg_b from table where avg_b>2 group by a;//错误
select a, avg(b) as avg_b from table group by a having avg_b>2;
//我们根据a分组聚合,并将聚合的列命名为avg_b。
//原表中没有avg_b,所以不能用where对avg_b进行行筛选
//只能用having,在筛选了列、聚合之后,再用having进行筛选行。
8、找出所有员工当前(to_date='9999-01-01')具体的薪水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 distinct salary
from salaries
where to_date='9999-01-01'
order by salary desc;
排序用order by,这里去重用的是distinct,也可以用group by。group by的写法在下面的其他答案中。
这里主要的问题是distince和group by的性能区别:
1)当重复量非常巨大,如1000w条中有300w条重复数据,用distinct比较好。
2)当重复量较小,如1000w条中有1w条重复数据,用group by比较好。
其他答案
select salary
from salaries
where to_date='9999-01-01'
group by salary
order by salary desc;
9、获取所有部门当前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));
答案
select de.dept_no, de.emp_no, sa.salary
from dept_manager as de
join salaries as sa
on de.emp_no=sa.emp_no
and de.to_date='9999-01-01'
and sa.to_date='9999-01-01';
此题讨论区有个争论点是,有人认为一个人可能同时任两个部门的领导,这里我们假设只能在一个部门工作,也就是如果换了部门,则to_date会变为当前时间而不是`9999-01-01`。
其他答案
select de.dept_no, de.emp_no, sa.salary
from dept_manager as de
join salaries as sa
on de.emp_no=sa.emp_no
where de.to_date='9999-01-01'
and sa.to_date='9999-01-01';
将时间筛选放到where中也可以通过。
10、获取所有非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));
答案
select emp_no
from employees
where emp_no not in
(select distinct emp_no
from dept_manager
where to_date='9999-01-01');
这题主要考察'not in'。
结尾
如果您发现我的文章有任何错误,或对我的文章有什么好的建议,请联系我!如果您喜欢我的文章,请点喜欢~*我是蓝白绛,感谢你的阅读!