牛客网SQL在线编程(第一天)

题目(2)

(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 order by hire_date desc limit 1 offset 2;

总结

查询排名第几的结果时,先进行 order by 排序操作,再利用 limit……offset 进行选取操作,limit 后跟读取的数据条数,offset 后跟跳过的数据条数。

题目(3)

(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 s.*,dm.dept_no
from salaries as s left join dept_manager as dm 
on s.emp_no = dm.emp_no 
where s.to_date = '9999-01-01' and dm.to_date = '9999-01-01';

总结

在解题过程中,一开始将 where 里面的条件错写成了 to_date = '9999-01-01',而两个表格中都含有 to_data 这一列,所以要分别指明都符合条件。并不是表格合并后再进行判断where 里面的条件。

题目(4)

(4)查找所有已经分配部门的员工的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 e.last_name,e.first_name,de.dept_no
from dept_emp as de left join employees as e
on de.emp_no = e.emp_no;

总结

由题目可知,需要查找所有已经分配部门的员工,也就是在dept_emp表中有emp_no的员工,所以,这里我使用的以dept_emp表为基础的左联结。同时这个问题也可以直接使用内联结inner join

题目(5)

(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 e.last_name,e.first_name,de.dept_no
from employees as e left join dept_emp as de
on e.emp_no = de.emp_no;

总结

题目中明确指出:也包括暂时没有分配具体部门的员工,所以很自然的就想到左联结(left join),以包含所有员工的数据表employees为基础,并放在左边。

题目(6)

(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 e.emp_no,s.salary
from employees as e left join salaries as s
on e.emp_no = s.emp_no and e.hire_date = s.from_date
order by e.emp_no desc;

总结

题目中指出:入职时候的薪水情况,则要考虑e.hire_date = s.from_date
而且,在一条SQL语句中包含多个表时,使用排序order by需要指明对应的数据表。

题目(7)

(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 s1.emp_no,count(s1.emp_no) as t
from salaries as s1, salaries as s2
where s1.to_date = s2.from_date and s2.salary > s1.salary
group by s1.emp_no having count(s1.emp_no)>15;
  • 这样的解答才可以过
select emp_no,count(emp_no) as t
from salaries 
group by emp_no having count(emp_no)>15;

总结

在这个题中,将涨幅为负值的也算作是涨薪一次。思路主要是先分组,然后对分组后的数据进行过滤(having)。

题目(8)

(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;

总结

做题时,一开始错写成了 distinct select,一定要记住,查询语句以select开头。

题目(9)

(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 dm.dept_no,dm.emp_no,s.salary
from dept_manager as dm left join salaries as s
on dm.emp_no = s.emp_no
where dm.to_date='9999-01-01' and s.to_date='9999-01-01';

总结

题目中要获取所有部门当前manager的当前薪水情况,所以以dept_manager 为基础进行左联结(left join)。需要注意的是,两个表中的to_date必须同时满足当前的条件。

题目(10)

(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 e.emp_no
from employees as e
where not e.emp_no in (select dm.emp_no from dept_manager as dm);

总结

本次解答中的子查询语句(select dm.emp_no from dept_manager as dm)不能用dept_manager.emp_no来代替。记住:只要是查询数据,必须使用查询语句

你可能感兴趣的:(刷题,数据分析,MySQL)