天天看点

MySQL学习之分组查询与连接查询

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;