美文网首页MySQL数据库SQL实战
数据库SQL实战|SQL答案集合及解析(1-10)

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

作者: 蓝白绛 | 来源:发表于2019-04-14 01:01 被阅读28次

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

    结尾

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

    相关文章

      网友评论

        本文标题:数据库SQL实战|SQL答案集合及解析(1-10)

        本文链接:https://www.haomeiwen.com/subject/ywymiqtx.html