数据库SQL实战|SQL答案集合及解析(1-10)

牛客数据库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 joinleft joinright 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'。

结尾

如果您发现我的文章有任何错误,或对我的文章有什么好的建议,请联系我!如果您喜欢我的文章,请点喜欢~*我是蓝白绛,感谢你的阅读!

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,968评论 6 482
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,601评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 153,220评论 0 344
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,416评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,425评论 5 374
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,144评论 1 285
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,432评论 3 401
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,088评论 0 261
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,586评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,028评论 2 325
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,137评论 1 334
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,783评论 4 324
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,343评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,333评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,559评论 1 262
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,595评论 2 355
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,901评论 2 345

推荐阅读更多精彩内容