1、分组查询
#分组查询
/*
语法:select 分组函数,列(要求出现在group by的后面)
from 表
【where 筛选条件】
group by 分组的列表
【order by字句】
特点:
分组前的筛选,操作原始表,在group by字句的前面,用where
分组后的筛选,操作的是分组后的结果集,在group by字句的后面,用having
①分组函数做筛选条件,放在having字句后
②最好使用分组前筛选
*/
#查询每个工种最高工资
SELECT MAX(salary),job_id
FROM employees
GROUP BY job_id;
#添加筛选条件,查询邮箱中包含a字母的,每个部门的平均工资、
SELECT AVG(salary),department_id
FROM employees
WHERE email LIKE '%a%'
GROUP BY department_id;
#添加分组后的筛选,查询那个部门员工个数大于2
SELECT COUNT(*),department_id
FROM employees
GROUP BY department_id
HAVING COUNT(*)>2;
#查询每个工种有奖金的员工的最高工资》12000的工作编号和最高工资
SELECT MAX(salary),job_id
FROM employees
WHERE commission_pct IS NOT NULL
GROUP BY job_id
HAVING MAX(salary)>12000;
#按员工姓名长度分组,查询每一组的员工个数,帅选员工个数》5的有哪些
SELECT COUNT(*) ,LENGTH(last_name)
FROM employees
GROUP BY LENGTH(last_name)
HAVING COUNT(*)>5;
#按多个字段分组,查询每个部门每个工种的平均工资
SELECT AVG(salary),department_id,job_id
FROM employees
GROUP BY job_id,department_id;
#按多个字段分组,查询每个部门每个工种的平均工资,并按高低显示
SELECT AVG(salary),department_id,job_id
FROM employees
WHERE department_id IS NOT NULL
GROUP BY job_id,department_id
HAVING AVG(salary)>10000
ORDER BY AVG(salary);
2、连接查询
#sql92语法
#连接查询
/*
添加有效的连接
按功能分类:
内连接:等值连接
非等值连接
自连接
外连接:左外连接
右外连接
全外连接
交叉连接
*/
/*
等值连接
①多表连接相当于多表的交集部分
②n表连接,至少需要n-1个连接条件
③多表顺序无要求
*/
#1、查询员工名和对应的部门名
SELECT last_name,department_name
FROM employees,departments
WHERE employees.`department_id`=departments.`department_id`;
#2、查询员工名,工种号、工种名
#如果为表起了别名,则查询字段不能使用原来的表名
SELECT employees.`last_name`,employees.`job_id`,jobs.`job_title`
FROM employees,jobs
WHERE employees.`job_id`=jobs.`job_id`;
#3、可以加筛选
SELECT last_name,department_name,commission_pct
FROM employees e,departments d
WHERE e.`department_id`=d.`department_id`
AND e.`commission_pct` IS NOT NULL;
#4、查询每个城市的部门个数(加分组)
SELECT COUNT(*) 个数,city
FROM departments d, locations l
WHERE d.`location_id`=l.`location_id`
GROUP BY city;
#5、查询每个工种的工种名和员工个数,并且按员工个数降序
SELECT job_title,COUNT(*)
FROM employees e,jobs j
WHERE e.`job_id`=j.`job_id`
GROUP BY job_title
ORDER BY COUNT(*);
/*
非等值连接:例如使用between and句式
*/
/*
自连接
*/
SELECT e.employee_id,e.last_name,m.employee_id,m.last_name
FROM employees e,employees m
WHERE e.`manager_id`=m.`employee_id`;
#sql99语法
/*
语法:
select 查询列表
from 表1【连接类型】
join 表2
on 连接条件
【where 筛选条件】
【group by 分组】
【having 分组】
【order by 排序列表】
【连接类型】分类:
内连接:inner
外连接
左外:left【outer】
右外:right【outer】
全外:full【outer】
交叉连接:cross
*/
#1、等值连接
SELECT last_name,department_name
FROM employees e
INNER JOIN departments d
ON e.`department_id`=d.`department_id`;
SELECT last_name,department_name,job_title
FROM employees e
INNER JOIN departments d ON e.`department_id`=d.`department_id`
INNER JOIN jobs j ON e.`job_id`=j.`job_id`
ORDER BY department_name DESC;
#2、非等值连接
/*
select salary,grade_level
from employees e
join job_grades g
on e.`salary` between g.'lowest_salary' and g.'higheat_sal';
*/
#3、自连接
SELECT e.employee_id,e.last_name,m.employee_id,m.last_name
FROM employees e
JOIN employees m
ON e.`manager_id`=m.`employee_id`;
#二、外连接
/*
特点:
1、外连接的查询结果为主表中的所有记录
如果从表中有和他匹配的,则显示匹配的值
如果从表中没有和他匹配的,则显示null
外连接的查询结果=内连接的查询结果+主表中有而从表中没有的记录
2、左外连接,left join 左边的是主表
右外连接,right join 右边的是主表
3、左外和右外交换两个表的顺序,可以实现同样的效果
4、全外连接=内连接结果+表1中有但表2中没有+表2中有但表1中没有
5、交叉连接=笛卡尔乘积
*/
#左外:查询哪个部门没有员工
SELECT d.*,e.employee_id
FROM departments d
LEFT OUTER JOIN employees e
ON d.`department_id`=e.`department_id`
WHERE e.`department_id`IS NULL;