MySQL学习笔记

MySQL学习笔记

目录

一、数据库的相关概念
二、初识MySQL
MySQL产品的介绍
MySQL产品的安装 ★
MySQL服务的启动和停止 ★
MySQL服务的登录和退出 ★
MySQL的常见命令和语法规范
五、DQL语言的学习 ★
基础查询 ★
条件查询 ★
排序查询 ★
常见函数 ★
分组函数 ★
分组查询 ★
连接查询 ★
子查询 √
分页查询 ★
union联合查询 √

六、DML语言的学习 ★
插入语句
修改语句
删除语句
七、DDL语言的学习
库和表的管理 √
常见数据类型介绍 √
常见约束 √
八、TCL语言的学习
事务和事务处理
九、视图的讲解 √
十、变量
十一、存储过程和函数
十二、流程控制结构

一、了解数据库

**数据库:**是按照数据结构来组织、存储和管理数据的仓库。

数据库的好处/优势:

  • 持久化数据到本地
  • 可以实现结构化查询,方便管理
  • 数据共享

数据库相关概念

  • DB:数据库,是按照数据结构来组织、存储和管理数据的仓库。
  • DBMS:数据库管理系统,用于创建、管理DB
  • RDBMS:关系型数据库管理系统,关系型可以理解为“表格”的概念。是基于关系模型的DBMS。
  • SQL:结构化查询语言,用于和DBMS通信的语言

关系型数据库管理系统的特点

  • 数据以表格的形式出现
  • 每行为各种记录名称
  • 每列为记录名称所对应的数据域
  • 许多的行和列组成一张表单
  • 若干的表单组成数据库

关系型数据库中的表格结构

image-20210303181744660

  • 键(key): 键的值在当前列中具有唯一性。
  • 值(value): 行的具体信息, 每个值必须与该列的数据类型相同;

关系型数据库存储数据的特点

  1. 数据放入表中,表再放到数据库中;
  2. 一个数据库中可以有多个表,每个表都有一个的名字,用来标识自己。表名具有唯一性;
  3. 表具有一些特性,这些特性定义了数据在表中如何存储,类似java中 “类”的设计;
  4. 表由列组成,我们也称为字段。所有表都是由一个或多个列组成的,每一列类似java 中的”属性”;
  5. 表中的数据是按行存储的,每一行类似于java中的“对象”。

二、初识MySQL

MySQL简介

  • MySQL隶属于MySQL AB公司,总部位于瑞典,后被oracle收购。

  • MySQL具体来说是RDBMS,当然了,RDBMS也是一种DBMS。

  • MySQL是基于客户机——服务器(C/S)的DBMS

DBMS分为两类:
-   基于共享文件系统的DBMS(Access)
-   基于客户机——服务器(C/S)DBMS(MySQL、Oracle、SqlServer...)
  • 官网地址:https://www.mysql.com/

MySQL优点

  • 成本低:开源
  • 性能高:执行快
  • 简单:易安装使用

MySQL安装

MySQL服务的启动和停止

  • 方式一:计算机——右击管理——服务
  • 方式二:DOS窗口中通过管理员身份运行
    net start 服务名(启动服务)
    net stop 服务名(停止服务)

MySQL服务端的登录和登出

  • 方式一:通过mysql自带的客户端(只限于root用户)

  • 方式二:通过windows自带的客户端

    登录:
    mysql 【-h主机名 -P端口号 】-u用户名 -p密码

    退出:
    exit或ctrl+C

MySQL常见命令

1.查看当前所有的数据库
show databases;

2.打开指定的库
use 库名

3.查看当前库的所有表
show tables;

4.查看指定库的所有表
show tables from 库名;

5.创建表
create table 表名(
	列名 列类型,
	列名 列类型,
	...
);

6.查看表结构
desc 表名;

7.查看服务器的版本
方式一:登录到mysql服务端
select version();
方式二:没有登录到mysql服务端
mysql --version
或
mysql --V

8.显示表中所有数据
select * from 表名;显示表中的所有数据

MySQL的语法规范

1.不区分大小写,但建议关键字大写,表名、列名小写
2.每条命令最好用分号结尾
3.每条命令根据需要,可以进行缩进 或换行
4.注释
单行注释:#注释文字
单行注释:-- 注释文字
多行注释:/* 注释文字 */

图形化用户界面客户端

SQLyog:是一个易于使用的、快速而简洁的图形化管理MYSQL数据库的工具

安装(略)

三、MySQL的SQL语言

SQL语言的分类

  • DQL(Data Query Language):数据查询语言
    select
  • DML(Data Manipulate Language):数据操作语言
    insert 、update、delete
  • DDL(Data Define Languge):数据定义语言
    create、drop、alter
  • TCL(Transaction Control Language):事务控制语言
    commit、rollback

本文示例使用的数据

  • myemployees.sql : 创建一个名为myemployees的数据库,并且这个数据库下有四张表

链接:https://pan.baidu.com/s/1RlrqLVArMMRu5qxqbuExbQ
提取码:dj2y

  1. employees表

image-20210304111354246

  1. departments表

image-20210304112500460

  1. locations表

    image-20210304112746946

  2. jobs表

    image-20210304113000856

  • girls.sql

1.admin表

image-20210308171012866

2.beauty

image-20210308171056313

3.boys

image-20210308171128116

DQL语言

数据查询语言

基础查询

语法


select 查询列表 from 表名
特点:
1、查询列表可以是:表中的字段、常量值、表达式、函数
2、查询的结果是一个虚拟的表格

示例


#1.查询单个字段
SELECT last_name FROM employees;

#2.查询多个字段
SELECT last_name,salary,email FROM employees; 

#3.查询所有字段
SELECT * FROM employees;

#4.查询常量值
SELECT 100;
SELECT 'jack';

#5.查询表达式
SELECT 100%98;

#6.查询函数
SELECT VERSION();

#7.起别名
/*
作用:
①便于理解
②如果要查询的字段有重名,使用别名可以区分出来
*/
#方式一
SELECT 100%98 AS result;

#方式二
SELECT 100%98 result;

#8.去重

#查询员工表中的所有部门的编号
SELECT DISTINCT department_id FROM employees;

#9.+号的作用
/*
java中的+号:
①运算符,两个操作数都为数值型
②连接符,只要有一个操作数为字符串

mysql中的+号:
仅有一个功能:运算符

select 100+90; 两个操作数都为数值型,则做加法运算
select '123' + 90; 其中一方为字符型,则试图转换为数值型,如果转换成功则做加法运算。
select 'jack' + 90; 如果转换失败则将转换失败的字符串转换成0,再作加法运算
select null + 90;  只要其中一方为null,则结果为null
*/

#10.字符串拼接CONCAT()
#查询员工名和姓连接成一个字段,并显示为姓名
SELECT CONCAT(last_name,first_name) AS 姓名
FROM employees;

测试题

1. 下面的语句是否可以执行成功
select last_name , job_id , salary as sal
from employees;
2. 下面的语句是否可以执行成功
select * from employees;
3. 找出下面语句中的错误
select employee_id , last_name,
salary * 12 “ANNUAL SALARY”
from employees;
4. 显示表departments的结构,并查询其中的全部数据
5. 显示出表employees中的全部job_id(不能重复)
6. 显示出表employees的全部列,各个列之间用逗号连接,列头显示成OUT_PUT

测试题解答

1.可以
2.可以
3.
SELECT employee_id , last_name,
salary * 12 “ANNUAL SALARY”
FROM employees;
','改为','

4.
DESC departments;
SELECT * FROM departments;

5.
SELECT DISTINCT job_id FROM employees;

6.
①方法1
SELECT CONCAT(`employee_id`,',',`first_name`,',',`last_name`,',',
`email`,',',`phone_number`,',',`job_id`,',',`salary`,',',IFNULL(`commission_pct`,0),',',`manager_id`,',',`department_id`,',',`hiredate`) AS OUT_PUT
FROM employees;

②方法2
SELECT CONCAT_WS(',',`employee_id`,`first_name`,`last_name`,`email`,
`phone_number`,`job_id`,`salary`,`commission_pct`,`manager_id`,
`department_id`,`hiredate`) AS OUT_PUT
FROM employees;

注:两种方法的结果是有些不一致的

额外补充:
(1)concat函数和concat_ws函数
concat函数在连接字符串的时候,只要其中一个是NULL,那么将返回NULL
concat_ws函数在连接字符串的时候,只要有一个字符串不是NULL,就不会返回NULL。concat_ws函数需要指定分隔符。
(2)IFNULL()函数
IFNULL(A,B) 表示如果A为NULL,则用B表示,否则为其本身

条件查询

语法

select 查询列表
from 表名
where 筛选条件

执行顺序:
①fromwhereselect

分类(其实就是按条件类型分类)

  • 按条件表达式筛选

    条件运算符:> 、< 、=、 !=、 <>、 >=、 <=、 <=>

  • 按逻辑表达式筛选

    逻辑运算符
    &&、||、 !
    and、or、not

  • 模糊查询

    like
    not like
    between and
    not between and
    in
    not in
    is null
    is not null

示例

#一、按条件表达式筛选

#案例1:查询工资>12000的员工信息
SELECT  * FROM employees
WHERE salary>12000;

#案例2:查询部门编号不等于90号的员工名和部门编号
SELECT 
	last_name,
	department_id
FROM
	employees
WHERE
	department_id<>90;

#二、按逻辑表达式筛选

#案例1:查询工资z在10000到20000之间的员工名、工资以及奖金
SELECT
	last_name,
	salary,
	commission_pct
FROM
	employees
WHERE
	salary>=10000 AND salary<=20000;

#案例2:查询部门编号不是在90到110之间,或者工资高于15000的员工信息
SELECT
	*
FROM
	employees
WHERE
	NOT(department_id>=90 AND  department_id<=110) OR salary>15000;

#三、模糊查询

#1. like
/*
特点:
①一般和通配符搭配使用
	通配符:
	% 任意多个字符,包含0个字符
	_ 任意单个字符
②转义
	当字符串中本身含有%,_通配符时,我们需要转义
	一种方法是:通配符前加\
	另一种方法是:通过ESCAPE自定义通配符
*/

#案例1:查询员工名中包含字符a的员工信息

SELECT 
	*
FROM
	employees
WHERE
	last_name LIKE '%a%';#abc
#案例2:查询员工名中第三个字符为e,第五个字符为a的员工名和工资
SELECT
	last_name,
	salary
FROM
	employees
WHERE
	last_name LIKE '__n_l%';

#案例3:查询员工名中第二个字符为_的员工名

SELECT
	last_name
FROM
	employees
WHERE
	last_name LIKE '_\_%';
---------------------
SELECT
	last_name
FROM
	employees
WHERE
	last_name LIKE '_$_%' ESCAPE '$';


#2. BETWEEN AND
/*
①使用between and 可以提高语句的简洁度
②包含临界值
③两个临界值不要调换顺序
*/

#案例1:查询员工编号在100到120之间的员工信息

SELECT
	*
FROM
	employees
WHERE
	employee_id >= 120 AND employee_id<=100;
#----------------------
SELECT
	*
FROM
	employees
WHERE
	employee_id BETWEEN 120 AND 100;

#3. in
/*
含义:判断某字段的值是否属于in列表中的某一项
特点:
	①使用in提高语句简洁度
	②in列表的值类型必须一致或兼容
	③in列表中不支持通配符
	

*/
#案例:查询员工的工种编号是 IT_PROG、AD_VP、AD_PRES中的一个员工名和工种编号

SELECT
	last_name,
	job_id
FROM
	employees
WHERE
	job_id = 'IT_PROT' OR job_id = 'AD_VP' OR JOB_ID ='AD_PRES';


#------------------

SELECT
	last_name,
	job_id
FROM
	employees
WHERE
	job_id IN( 'IT_PROT' ,'AD_VP','AD_PRES');
	
#4、is null
/*
=或<>不能用于判断null值 ★★
is null或is not null 可以判断null值
*/

#案例1:查询没有奖金的员工名和奖金率
SELECT
	last_name,
	commission_pct
FROM
	employees
WHERE
	commission_pct IS NULL;


#案例1:查询有奖金的员工名和奖金率
SELECT
	last_name,
	commission_pct
FROM
	employees
WHERE
	commission_pct IS NOT NULL;


额外补充:安全等于  <=>

#案例1:查询没有奖金的员工名和奖金率
SELECT
	last_name,
	commission_pct
FROM
	employees
WHERE
	commission_pct <=>NULL;
	
#案例2:查询工资为12000的员工信息
SELECT
	last_name,
	salary
FROM
	employees

WHERE 
	salary <=> 12000;
	

#is null pk <=>

IS NULL: 仅仅可以判断NULL值,可读性较高,建议使用
<=>     : 既可以判断NULL值,又可以判断普通的数值,可读性较低

测试题

1. 查询工资大于12000的员工姓名和工资
2. 查询员工号为176的员工的姓名和部门号和年薪
3. 选择工资不在5000到12000的员工的姓名和工资
4. 选择在20或50号部门工作的员工姓名和部门号
5. 选择公司中没有管理者的员工姓名及job_id
6. 选择公司中有奖金的员工姓名,工资和奖金级别
7. 选择员工姓名的第三个字母是a的员工姓名
8. 选择姓名中有字母a和e的员工姓名
9. 显示出表employees表中 first_name 以 'e'结尾的员工信息
10. 显示出表employees部门编号在80-100之间 的姓名、职位
11. 显示出表employees的manager_id 是 100,101,110 的员工姓名、职位
12.试问:select * FROM employess;和
SELECT * FROM empployees WHERE commission_pct LIKE '%%' AND last_name LIKE '%%';
结果是否一样?并说明理由

测试题解答

1. 查询工资大于12000的员工姓名和工资
SELECT last_name, salary
FROM employees
WHERE salary > 12000;

2. 查询员工号为176的员工的姓名和部门号和年薪
SELECT last_name, `department_id`, `salary`*12*(1+IFNULL(`commission_pct`,0)) AS 年薪
FROM employees
WHERE `employee_id` = 176;

3. 选择工资不在500012000的员工的姓名和工资
SELECT last_name, salary
FROM employees
WHERE salary < 5000 OR salary > 12000;

4. 选择在2050号部门工作的员工姓名和部门号
SELECT last_name, department_id
FROM employees
WHERE department_id BETWEEN 20 AND 50;

5. 选择公司中没有管理者的员工姓名及job_id
SELECT last_name, job_id
FROM employees
WHERE manager_id IS NULL;

6. 选择公司中有奖金的员工姓名,工资和奖金级别
SELECT last_name, salary, commission_pct
FROM employees
WHERE commission_pct IS NOT NULL;

7. 选择员工姓名的第三个字母是a的员工姓名
SELECT last_name
FROM employees
WHERE last_name LIKE '__a%';

8. 选择姓名中有字母a和e的员工姓名
SELECT last_name
FROM employees
WHERE last_name LIKE '%a%' OR last_name LIKE '%e%';

9. 显示出表employees表中 first_name 以 'e'结尾的员工信息
SELECT *
FROM employees
WHERE first_name LIKE '%e';

10. 显示出表employees部门编号在80-100之间 的姓名、工种、部门编号
SELECT last_name, job_id, department_id
FROM employees
WHERE department_id BETWEEN 80 AND 100;

11. 显示出表employees的manager_id是 100,101,110 的员工姓名、工种、领导员工编号
SELECT last_name, job_id, manager_id
FROM employees
WHERE manager_id IN (100,101,110);

12.
不一样。
因为commission_pct和last_name可能包含了NULL
排序查询

语法

select 查询列表
fromwhere 筛选条件】
order by 排序列表 【asc|desc】

特点:
1asc:代表的是升序,可以省略
desc:代表的是降序
2order by子句可以支持 单个字段、别名、表达式、函数、多个字段
3order by子句在查询语句的最后面,除了limit子句

执行顺序:
①fromwhereselectorder by

示例

#1、按单个字段排序
#案例:查询员工信息,按工资从高到低排序
SELECT * FROM employees ORDER BY salary DESC;

#2、添加筛选条件再排序

#案例:查询部门编号>=90的员工信息,并按员工编号降序

SELECT *
FROM employees
WHERE department_id>=90
ORDER BY employee_id DESC;


#3、按表达式排序
#案例:查询员工信息 按年薪降序

SELECT *,salary*12*(1+IFNULL(commission_pct,0))
FROM employees
ORDER BY salary*12*(1+IFNULL(commission_pct,0)) DESC;


#4、按别名排序
#案例:查询员工信息 按年薪升序

SELECT *,salary*12*(1+IFNULL(commission_pct,0)) 年薪
FROM employees
ORDER BY 年薪 ASC;

#5、按函数排序
#案例:查询员工名,并且按名字的长度降序

SELECT LENGTH(last_name),last_name 
FROM employees
ORDER BY LENGTH(last_name) DESC;

#6、按多个字段排序

#案例:查询员工信息,要求先按工资降序,再按employee_id升序
SELECT *
FROM employees
ORDER BY salary DESC,employee_id ASC;

测试题

1. 查询员工的姓名和部门号和年薪,按年薪降序 按姓名升序
2. 选择工资不在800017000的员工的姓名和工资,按工资降序
3. 查询邮箱中包含e的员工信息,并先按邮箱的字节数降序,再按部门号升序

测试题解答

1. 查询员工的姓名和部门号和年薪,按年薪降序 按姓名升序
SELECT last_name, department_id, salary * 12 * (1 + IFNULL(commission_pct, 0)) 年薪
FROM employees
ORDER BY 年薪 DESC, last_name ASC;

2. 选择工资不在800017000的员工的姓名和工资,按工资降序
SELECT last_name, salary
FROM employees
WHERE salary NOT BETWEEN 8000 AND 17000
ORDER BY salary DESC

3. 查询邮箱中包含e的员工信息,并先按邮箱的字节数降序,再按部门号升序
SELECT * 
FROM employees
WHERE email LIKE '%e%'
ORDER BY LENGTH(email) DESC, department_id ASC;
常见函数
一、单行函数
	一个输入一个输出
	1、字符函数
		concat 拼接
		substr/substring 截取子串
		upper 转换成大写
		lower 转换成小写
		trim 去前后指定的空格和字符
		ltrim 去左边空格
		rtrim 去右边空格
		replace 替换
		lpad 左填充
		rpad 右填充
		instr 返回子串第一次出现的索引
		length  获取字节个数
		
	2、数学函数
		round 四舍五入
		rand 随机数
		floor 向下取整
		ceil向上取整
		mod 取余
		truncate 截断
	3、日期函数
		now 当前系统日期+时间
		curdate 当前系统日期
		curtime 当前系统时间
		str_to_date 将字符转换成日期
		date_format 将日期转换成字符
	4、流程控制函数
		if 处理双分支
		case语句 处理多分支
			情况1:处理等值判断
			情况2:处理条件判断
		
	5、其他函数
		version 版本
		database 当前库
		user 当前连接用户


二、分组函数
        又称统计函数、聚合函数、组函数
        多个输入一个输出
	sum 求和
	max 最大值
	min 最小值
	avg 平均值
	count 计数

	特点:
	1、以上五个分组函数都忽略null值,除了count(*)
	2、sum和avg一般用于处理数值型
		max、min、count可以处理任何数据类型
        3、都可以搭配distinct使用,用于统计去重后的结果
	4、count的参数可以支持:
		字段、*、常量值,一般放1

	   建议使用 count(*)

示例

#一、字符函数

#1.length 获取参数值的字节个数
SELECT LENGTH('john');   4
SELECT LENGTH('张三丰hahaha');  15

SHOW VARIABLES LIKE '%char%';  ——查看当前使用的字符集

utf-8 一个汉字占3个字节
GBK  一个汉字占2个字节

#2.concat 拼接字符串

SELECT CONCAT(last_name,'_',first_name) 姓名 FROM employees;

#3.upper、lower
SELECT UPPER('john');
SELECT LOWER('joHn');
#示例:将姓变大写,名变小写,然后拼接
SELECT CONCAT(UPPER(last_name),LOWER(first_name))  姓名 FROM employees;

#4.substr、substring
注意:索引从1开始
#截取从指定索引处后面所有字符
SELECT SUBSTR('李莫愁爱上了陆展元',7)  out_put;

#截取从指定索引处指定字符长度的字符
SELECT SUBSTR('李莫愁爱上了陆展元',1,3) out_put;


#案例:姓名中首字符大写,其他字符小写然后用_拼接,显示出来

SELECT CONCAT(UPPER(SUBSTR(last_name,1,1)),'_',LOWER(SUBSTR(last_name,2)))  out_put
FROM employees;

#5.instr 返回子串第一次出现的索引,如果找不到返回0

SELECT INSTR('杨不殷六侠悔爱上了殷六侠','殷八侠') AS out_put;

#6.trim

SELECT LENGTH(TRIM('    张翠山    ')) AS out_put;

SELECT TRIM('aa' FROM 'aaaaaaaaa张aaaaaaaaaaaa翠山aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa')  AS out_put;

#7.lpad 用指定的字符实现左填充指定长度

SELECT LPAD('殷素素',2,'*') AS out_put;

#8.rpad 用指定的字符实现右填充指定长度

SELECT RPAD('殷素素',12,'ab') AS out_put;


#9.replace 替换

SELECT REPLACE('周芷若周芷若周芷若周芷若张无忌爱上了周芷若','周芷若','赵敏') AS out_put;

------------------------------------------------------------------------------------------------------
#二、数学函数

#round 四舍五入
SELECT ROUND(-1.55);
SELECT ROUND(1.567,2);

#ceil 向上取整,返回>=该参数的最小整数

SELECT CEIL(-1.02);

#floor 向下取整,返回<=该参数的最大整数
SELECT FLOOR(-9.99);

#truncate 截断

SELECT TRUNCATE(1.69999,1);

#mod取余
/*
mod(a,b) :  a-a/b*b

mod(-10,-3):-10- (-10)/(-3)*(-3)=-1
*/
SELECT MOD(10,-3);
SELECT 10%3;

-----------------------------------------------------------------------------------------------------
#三、日期函数

#now 返回当前系统日期+时间
SELECT NOW();

#curdate 返回当前系统日期,不包含时间
SELECT CURDATE();

#curtime 返回当前时间,不包含日期
SELECT CURTIME();


#可以获取指定的部分,年、月、日、小时、分钟、秒
SELECT YEAR(NOW());
SELECT YEAR('1998-1-1');

SELECT  YEAR(hiredate)FROM employees;

SELECT MONTH(NOW());    数字
SELECT MONTHNAME(NOW());  英文


#str_to_date 将字符通过指定的格式转换成日期

SELECT STR_TO_DATE('1998-3-2','%Y-%c-%d') AS out_put;

#查询入职日期为1992--4-3的员工信息
SELECT * FROM employees WHERE hiredate = '1992-4-3';

SELECT * FROM employees WHERE hiredate = STR_TO_DATE('4-3 1992','%c-%d %Y');


#date_format 将日期转换成字符

SELECT DATE_FORMAT(NOW(),'%y年%m月%d日') AS out_put;

#查询有奖金的员工名和入职日期(xx月/xx日 xx年)
SELECT last_name,DATE_FORMAT(hiredate,'%m月/%d日 %y年') 入职日期
FROM employees
WHERE commission_pct IS NOT NULL;

-------------------------------------------------------------------------------------------------
#四、其他函数

SELECT VERSION();
SELECT DATABASE();
SELECT USER();

-------------------------------------------------------------------------------------------------

#五、流程控制函数
#1.if函数: if else 的效果

SELECT IF(10<5,'大','小');

SELECT last_name,commission_pct,IF(commission_pct IS NULL,'没奖金,呵呵','有奖金,嘻嘻') 备注
FROM employees;




#2.case函数的使用一: switch case 的效果

/*
java中
switch(变量或表达式){
	case 常量1:语句1;break;
	...
	default:语句n;break;


}

mysql中

case 要判断的字段或表达式
when 常量1 then 要显示的值1或语句1;
when 常量2 then 要显示的值2或语句2;
...
else 要显示的值n或语句n;
end
*/

/*案例:查询员工的工资,要求

部门号=30,显示的工资为1.1倍
部门号=40,显示的工资为1.2倍
部门号=50,显示的工资为1.3倍
其他部门,显示的工资为原工资

*/


SELECT salary 原始工资,department_id,
CASE department_id
WHEN 30 THEN salary*1.1
WHEN 40 THEN salary*1.2
WHEN 50 THEN salary*1.3
ELSE salary
END AS 新工资
FROM employees;



#3.case 函数的使用二:类似于 多重if
/*
java中:
if(条件1){
	语句1;
}else if(条件2){
	语句2;
}
...
else{
	语句n;
}

mysql中:

case 
when 条件1 then 要显示的值1或语句1
when 条件2 then 要显示的值2或语句2
。。。
else 要显示的值n或语句n
end
*/

#案例:查询员工的工资的情况
如果工资>20000,显示A级别
如果工资>15000,显示B级别
如果工资>10000,显示C级别
否则,显示D级别

SELECT salary,
CASE 
WHEN salary>20000 THEN 'A'
WHEN salary>15000 THEN 'B'
WHEN salary>10000 THEN 'C'
ELSE 'D'
END AS 工资级别
FROM employees;

测试题

  • 显示系统时间(注:日期+时间)
  • 查询员工号,姓名,工资,以及工资提高百分之20%后的结果(new salary)
  • 将员工的姓名按首字母排序,并写出姓名的长度(length)
  • 做一个查询,产生下面的结果

image-20210305184258876

  • 使用case-when,按照下面的条件:
    job grade
    AD_PRES A
    ST_MAN B
    IT_PROG C
    SA_REP D
    ST_CLERK E
    产生下面的结果

    image-20210305184517771

测试题解答

1.
SELECT NOW();

2.
SELECT `employee_id`,`salary`,`salary`*(1 + 0.2) new_salary
FROM `employees`;

3.
SELECT last_name,LENGTH(last_name) 'length'
FROM `employees`
ORDER BY SUBSTR(last_name,1,1);

4.
SELECT CONCAT(last_name, ' earns ' , salary, ' monthly but wants ', salary*3 ) AS 'Dream Salary'
FROM `employees`;

5.
SELECT last_name, job_id, 
CASE `job_id`
WHEN 'AD_PRES' THEN 'A'
WHEN 'ST_MAN' THEN 'B'
WHEN 'IT_PROG' THEN 'C'
WHEN 'SA_REP' THEN 'D'
WHEN 'ST_CLERK' THEN 'E'
ELSE job_id
END AS Grade
FROM employees;
分组函数
功能:用作统计使用,又称为聚合函数或统计函数或组函数

分类:
sum 求和、avg 平均值、max 最大值 、min 最小值 、count 计算个数

特点:
1、sum、avg一般用于处理数值型
   max、min、count可以处理任何类型
   
2、以上分组函数都忽略null3、可以和distinct搭配实现去重的运算

4、count函数的单独介绍
一般使用count(*)用作统计行数

5、和分组函数一同查询的字段要求是group by后的字段

示例

#1、简单 的使用
SELECT SUM(salary) FROM employees;
SELECT AVG(salary) FROM employees;
SELECT MIN(salary) FROM employees;
SELECT MAX(salary) FROM employees;
SELECT COUNT(salary) FROM employees;


SELECT SUM(salary),AVG(salary) 平均,MAX(salary) 最高,MIN(salary) 最低,COUNT(salary) 个数
FROM employees;

SELECT SUM(salary),ROUND(AVG(salary),2) 平均,MAX(salary) 最高,MIN(salary) 最低,COUNT(salary) 个数
FROM employees;

#2、参数支持哪些类型

SELECT SUM(last_name) ,AVG(last_name) FROM employees;
SELECT SUM(hiredate) ,AVG(hiredate) FROM employees;

SELECT MAX(last_name),MIN(last_name) FROM employees;

SELECT MAX(hiredate),MIN(hiredate) FROM employees;

SELECT COUNT(commission_pct) FROM employees;
SELECT COUNT(last_name) FROM employees;

#3、是否忽略null

SELECT SUM(commission_pct) ,AVG(commission_pct),SUM(commission_pct)/35,SUM(commission_pct)/107 FROM employees;

SELECT MAX(commission_pct) ,MIN(commission_pct) FROM employees;

SELECT COUNT(commission_pct) FROM employees;
SELECT commission_pct FROM employees;


#4、和distinct搭配

SELECT SUM(DISTINCT salary),SUM(salary) FROM employees;
SELECT COUNT(DISTINCT salary),COUNT(salary) FROM employees;



#5、count函数的详细介绍

SELECT COUNT(salary) FROM employees;
SELECT COUNT(*) FROM employees;
SELECT COUNT(1) FROM employees;

效率:
MYISAM存储引擎下  ,COUNT(*)的效率高
INNODB存储引擎下,COUNT(*)COUNT(1)的效率差不多,比COUNT(字段)要高一些


#6、和分组函数一同查询的字段有限制

SELECT AVG(salary),employee_id  FROM employees;

测试

1.查询公司员工工资的最大值,最小值,平均值,总和
2.查询员工表中的最大入职时间和最小入职时间的相差天数 (DIFFRENCE)
3.查询部门编号为90的员工个数

测试题解答

1.
SELECT MAX(salary) 最大值,MIN(salary) 最小值,AVG(salary) 平均值,SUM(salary)FROM employees;

2.
SELECT MAX(hiredate) 最大,MIN(hiredate) 最小,(MAX(hiredate)-MIN(hiredate))/1000/3600/24 DIFFRENCE
FROM employees;

SELECT DATEDIFF(MAX(hiredate),MIN(hiredate)) DIFFRENCE
FROM employees;

SELECT DATEDIFF('1995-2-7','1995-2-6');

3.
SELECT COUNT(*) FROM employees WHERE department_id = 90;
分组查询

语法

select 查询列表
from 表
【where 筛选条件】
group by 分组的字段
【having 筛选条件】
【order by 排序的字段】

特点:
1、和分组函数一同查询的字段必须是group by后出现的字段
2、筛选分为两类:分组前筛选和分组后筛选
		        针对的表		         位置		        连接的关键字
分组前筛选	  原始表			      group bywhere
分组后筛选	  group by后的结果集    	 group byhaving

问题1:分组函数做筛选能不能放在where后面
答:不能

问题2where——group by——having
一般来讲,能用分组前筛选的,尽量使用分组前筛选,提高效率

3、分组可以按单个字段也可以按多个字段(无顺序要求),表达式或函数(用的较少)
4、可以搭配着排序使用


执行顺序:
①fromwheregroup byhavingselectorder by

示例

#1.简单的分组

#案例1:查询每个工种的员工平均工资
SELECT AVG(salary),job_id
FROM employees
GROUP BY job_id;

#案例2:查询每个位置的部门个数
SELECT COUNT(*),location_id
FROM departments
GROUP BY location_id;

----------------------------------------------------------------------------------------------------------
#2、可以实现分组前的筛选

#案例1:查询邮箱中包含a字符的 每个部门的最高工资
SELECT MAX(salary),department_id
FROM employees
WHERE email LIKE '%a%'
GROUP BY department_id;

#案例2:查询有奖金的每个领导手下员工的平均工资

SELECT AVG(salary),manager_id
FROM employees
WHERE commission_pct IS NOT NULL
GROUP BY manager_id;

--------------------------------------------------------------------------------------------------------
#3、分组后筛选

#案例1:查询哪个部门的员工个数>5

#①查询每个部门的员工个数
SELECT COUNT(*),department_id
FROM employees
GROUP BY department_id;

#② 筛选刚才①结果

SELECT COUNT(*),department_id
FROM employees
GROUP BY department_id
HAVING COUNT(*)>5;

#案例2:每个工种有奖金的员工的最高工资>12000的工种编号和最高工资
SELECT job_id,MAX(salary)
FROM employees
WHERE commission_pct IS NOT NULL
GROUP BY job_id
HAVING MAX(salary)>12000;

#案例3:领导编号>102的每个领导手下的最低工资大于5000的领导编号和最低工资
manager_id>102

SELECT manager_id,MIN(salary)
FROM employees
WHERE manager_id > 102
GROUP BY manager_id
HAVING MIN(salary)>5000;

------------------------------------------------------------------------------------------------
#4.添加排序

#案例:每个工种有奖金的员工的最高工资>6000的工种编号和最高工资,按最高工资升序
SELECT job_id, MAX(salary) m
FROM employees
WHERE commission_pct IS NOT NULL
GROUP BY job_id
HAVING m>6000
ORDER BY m ;

---------------------------------------------------------------------------------------------------
#5.按多个字段分组

#案例:查询每个工种每个部门的最低工资,并按最低工资降序
SELECT MIN(salary),job_id,department_id
FROM employees
GROUP BY department_id,job_id
ORDER BY MIN(salary) DESC;

同时满足department_id、job_id相同的为一组

测试

1. 查询各job_id的员工工资的最大值,最小值,平均值,总和,并按job_id升序
2. 查询员工最高工资和最低工资的差距(DIFFERENCE)
3. 查询各个管理者手下员工的最低工资,其中最低工资不能低于6000,没有管理者的员工不计算在内
4. 查询所有部门的编号,员工数量和工资平均值,并按平均工资降序
5. 选择具有各个job_id的员工人数

测试题解答

1. 查询各job_id的员工工资的最大值,最小值,平均值,总和,并按job_id升序
SELECT job_id, MAX(`salary`) 最大值, MIN(`salary`) 最小值, AVG(`salary`) 平均值, SUM(`salary`) 总和
FROM employees
GROUP BY job_id
ORDER BY job_id;

2. 查询员工最高工资和最低工资的差距(DIFFERENCE)
SELECT MAX(`salary`) - MIN(`salary`) AS DIFFERENCE
FROM `employees`;


3. 查询各个管理者手下员工的最低工资,其中最低工资不能低于6000,没有管理者的员工不计算在内
SELECT `manager_id`, MIN(`salary`) 员工最低工资
FROM employees
WHERE `manager_id` IS NOT NULL
GROUP BY manager_id
HAVING 员工最低工资 >= 6000


4. 查询所有部门的编号,员工数量和工资平均值,并按平均工资降序
SELECT `department_id` 部门编号, COUNT(*) 员工数量, AVG(`salary`) 平均工资
FROM employees
GROUP BY department_id
ORDER BY 平均工资 DESC;

5. 选择具有各个job_id的员工人数
SELECT job_id, COUNT(job_id) 员工人数
FROM `employees`
GROUP BY job_id;
连接查询

介绍

含义:又称多表查询,当查询的字段来自于多个表时,就会用到连接查询

笛卡尔乘积现象:表1 有m行,表2有n行,结果=m*n行

发生原因:没有有效的连接条件
如何避免:添加有效的连接条件

分类:

	按年代分类:
	sql92标准:仅仅支持内连接
	sql99标准【推荐】:支持内连接+外连接(左外和右外)+交叉连接
	
	按功能分类:
		内连接:
			等值连接
			非等值连接
			自连接
		外连接:
			左外连接
			右外连接
			全外连接
		
		交叉连接
sql92标准
sql92标准仅仅支持内连接,而内连接包括了:等值连接、非等值连接、自连接
#1、等值连接(条件为=)
/*
① 多表等值连接的结果为多表的交集部分
②n表连接,至少需要n-1个连接条件
③ 多表的顺序没有要求
④一般需要为表起别名
⑤可以搭配前面介绍的所有子句使用,比如排序、分组、筛选
*/

①简单等值连接

#案例1:查询女神名和对应的男神名
SELECT NAME,boyName 
FROM boys,beauty
WHERE beauty.boyfriend_id= boys.id;

#案例2:查询员工名和对应的部门名
SELECT last_name,department_name
FROM employees,departments
WHERE employees.`department_id`=departments.`department_id`;

②为表起别名
/*
①提高语句的简洁度
②区分多个重名的字段

注意:如果为表起了别名,则查询的字段就不能使用原来的表名去限定

*/
#查询员工名、工种号、工种名

SELECT e.last_name,e.job_id,j.job_title
FROM employees  e,jobs j
WHERE e.`job_id`=j.`job_id`;


#③两个表的顺序调换(结果相同)

#查询员工名、工种号、工种名

SELECT e.last_name,e.job_id,j.job_title
FROM jobs j,employees e
WHERE e.`job_id`=j.`job_id`;

#④可以加筛选

#案例1:查询有奖金的员工名、部门名

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;

#案例2:查询城市名中第二个字符为o的部门名和城市名

SELECT department_name,city
FROM departments d,locations l
WHERE d.`location_id` = l.`location_id`
AND city LIKE '_o%';

#⑤可以加分组

#案例1:查询每个城市的部门个数

SELECT COUNT(*) 个数,city
FROM departments d,locations l
WHERE d.`location_id`=l.`location_id`
GROUP BY city;

#案例2:查询有奖金的每个部门的部门名和部门的领导编号和该部门的最低工资
SELECT department_name,d.`manager_id`,MIN(salary)
FROM departments d,employees e
WHERE d.`department_id`=e.`department_id`
AND commission_pct IS NOT NULL
GROUP BY department_name,d.`manager_id`;

#⑥可以加排序

#案例:查询每个工种的工种名和员工的个数,并且按员工个数降序

SELECT job_title,COUNT(*)
FROM employees e,jobs j
WHERE e.`job_id`=j.`job_id`
GROUP BY job_title
ORDER BY COUNT(*) DESC;

#⑦可以实现三表连接?

#案例:查询员工名、部门名和所在的城市

SELECT last_name,department_name,city
FROM employees e,departments d,locations l
WHERE e.`department_id`=d.`department_id`
AND d.`location_id`=l.`location_id`
AND city LIKE 's%'
ORDER BY department_name DESC;

----------------------------------------------------------------------------------------------
#2、非等值连接(非等式条件,如between...and)
SELECT salary,grade_level
FROM employees e,job_grades g
WHERE salary BETWEEN g.`lowest_sal` AND g.`highest_sal`----------------------------------------------------------------------------------------------
#3、自连接(特殊的等值连接,自己连接自己)

#案例:查询 员工名和上级的名称

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

测试

1.显示所有员工的姓名,部门号和部门名称
SELECT e.last_name, e.`department_id`, d.`department_name`
FROM employees e,departments d
WHERE e.`department_id` = d.`department_id`;

2.查询90号部门员工的job_id和90号部门的location_id
SELECT e.job_id, d.location_id
FROM `departments` d,`employees` e
WHERE e.`department_id` = d.`department_id` AND e.`department_id` = 90;

3.选择所有有奖金的员工的last_name,department_name,location_id,city
SELECT last_name,department_name,d.`location_id`,city
FROM `employees` e,`departments` d,`locations` l
WHERE e.`department_id` = d.`department_id` AND d.`location_id` = l.`location_id`
AND e.`commission_pct` IS NOT NULL;

4.选择city在Toronto工作的员工的last_name,job_id,department_id,department_name
SELECT last_name,job_id,d.department_id,department_name
FROM `employees` e, `departments` d, `locations` l
WHERE e.`department_id` = d.`department_id` AND d.`location_id` = l.`location_id`
AND l.`city` = 'Toronto';

5.查询每个工种、每个部门的部门名、工种名和最低工资
SELECT `department_name`, `job_title`, MIN(`salary`) 最低工资
FROM  `departments` d, `jobs` j, `employees` e
WHERE d.`department_id` = e. `department_id` AND e.`job_id` = j.`job_id`
GROUP BY j.job_id, d.department_id;

6.查询每个国家下的部门数大于2的国家编号
SELECT `country_id`,COUNT(*) 部门数
FROM `departments` d,`locations` l
WHERE d.`location_id` = l.`location_id`
GROUP BY l.country_id
HAVING 部门数 > 2;

7.选择指定员工的姓名,员工号,以及他的管理者的姓名和员工号,结果类似于下面的格式
employees  Emp#  manager  Mgr#
kochhar    101   king     100

SELECT e.last_name employees, e.employee_id "Emp#",m.last_name 管理者, m.employee_id "Mgr#"
FROM employees e, employees m
WHERE e.manager_id = m.employee_id
AND e.last_name = 'Kochhar';

sql99标准
sql99标准【推荐】:支持内连接+外连接(左外和右外)+交叉连接

语法:
	select 查询列表
	from1 别名 【连接类型】
	join2 别名 on 连接条件
	【where 筛选条件】
	【group by 分组】
	【having 筛选条件】
	【order by 排序列表】

内连接:inner
外连接
	左外:leftouter】
	右外:rightouter】
	全外:fullouter】
交叉连接:cross

-----------------------------------------------------------------------
一)内连接
语法:
select 查询列表
from1 别名
【innerjoin2 别名
on 连接条件;

分类:
等值
非等值
自连接

特点:


1、等值连接
#案例1.查询员工名、部门名
select last_name, department_name
from departments d
inner join employees e
on e.`department_id` = d.`department_id`;

#案例2.查询名字中包含e的员工名和工种名(添加筛选)
select e.last_name, j.`job_title`
from employees e
inner join jobs j
on e.`job_id` = j.`job_id`
where e.last_name like '%e%';

#案例3.查询部门个数>3的城市名和部门个数,(添加分组 + 筛选)
select `city`,count(*) 部门个数
from `departments` d
inner join `locations` l
on l.`location_id` = d.`location_id`
group by l.`city`
having 部门个数 > 3;

#案例4.查询哪个部门的员工个数>3的部门名和员工个数,并按个数降序(添加排序)
select d.`department_name`,count(*) 员工个数
from `departments` d
inner join `employees` e
on e.`department_id` = d.`department_id`
group by d.`department_name`
having 员工个数 > 3
order by 员工个数 desc;

#案例5.查询员工名、部门名、工种名,并按部门名降序(添加三表连接)
select e.last_name, d.`department_name`, j.`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 d.`department_name` desc;


2.非等值连接
****这里的示例使用的数据没找到,所以自己建了个表****
CREATE TABLE IF NOT EXISTS `job_grades`(
	`grade_level` VARCHAR(3),
	`lowest_sal` INT(11),
	`highest_sal` INT(11)
);
并插入了一些数据
INSERT INTO job_grades VALUES('A',2000,3000);
INSERT INTO job_grades VALUES('B',3000,4000);
INSERT INTO job_grades VALUES('C',4000,5000);
INSERT INTO job_grades VALUES('D',5000,6000);
INSERT INTO job_grades VALUES('E',6000,7000);
INSERT INTO job_grades VALUES('F',7000,10000);
INSERT INTO job_grades VALUES('G',10000,50000);

#查询员工的工资级别
SELECT salary, grade_level
FROM employees e
JOIN job_grades g
ON e.`salary` BETWEEN g.`lowest_sal` AND g.`highest_sal`;

#查询工资级别的个数>20的个数,并且按工资级别降序
SELECT COUNT(*) 个数, grade_level
FROM employees e
JOIN job_grades j
ON e.`salary` BETWEEN j.`lowest_sal` AND j.`highest_sal`
GROUP BY grade_level
HAVING 个数>20
ORDER BY grade_level;

3.自连接
#查询员工名字、上级名字
SELECT e.last_name, m.last_name
FROM employees e
JOIN employees m
ON e.`manager_id` = m.`employee_id`;

#查询姓名中包含字符k的员工的名字、上级名字
SELECT e.last_name, m.last_name
FROM employees e
JOIN employees m
ON e.`manager_id` = m.`employee_id`
WHERE e.last_name LIKE '%k%';

--------------------------------------------------------------------
二)外连接
/*
应用场景:用于查询一个表中有,另一个表没有的记录
特点:
1.外连接的查询结果为主表中的所有记录
  如果从表中有和它匹配的,则显示匹配的值
  如果从表中没有和它匹配的,则显示null
  外连接查询结果=内连接查询结果+主表中有而从表中没有的记录
 
2.左外连接,left左边的是主表
  右外连接,right右边的是主表
  
3.左外连接和右外连接交换两个表的顺序,可以实现同样的效果
4.全外连接 = 内连接的结果 + 表1中有但表2没有的 + 表2中有但表1没有的
 */
 
 查询男朋友 不在男神表的女神名
 SELECT b.name
 FROM beauty b
 LEFT JOIN boys bo
 ON b.`boyfriend_id` = bo.`id`
 WHERE bo.`id` IS NULL;
 
#案例1:查询哪个部门没有员工
#左外
SELECT d.`department_name`
FROM departments d
LEFT JOIN employees e
ON d.`department_id` = e.`department_id`
WHERE e.`employee_id` IS NULL;

#右外
SELECT d.`department_name`
FROM employees e
RIGHT JOIN departments d
ON d.`department_id` = e.`department_id`
WHERE e.`employee_id` IS NULL;

#全外(注意:这里是举例,mysql并不支持全外语法)
USE girls;
SELECT b.*, bo.*
FROM beauty b
FULL JOIN boys bo
ON b.`boyfriend_id` = bo.`id`;

三)交叉连接
SELECT b.*,bo.*
FROM beauty b
CROSS JOIN boys bo;

测试

1.查询编号>3的女神的男朋友信息,如果有则列出详细,如果没有,用NULL填充
SELECT b.id, b.name, bo.*
FROM beauty b
LEFT JOIN boys bo
ON b.`boyfriend_id` = bo.`id`
WHERE b.`id` > 3

2.查询哪个城市没有部门
SELECT l.`city`
FROM `locations` l
LEFT JOIN `departments` d
ON l.`location_id` = d.`location_id`
WHERE d.`department_id` IS NULL;

3.查询部门名为SAL或IT的员工信息
SELECT e.*,d.`department_name`
FROM employees e
LEFT JOIN `departments` d
ON e.`department_id` = d.`department_id`
WHERE d.`department_name` IN ('SAL','IT');
子查询

概念

子查询
含义:
出现在其他语句中的select语句,称为子查询或内查询外部的查询语句,称为主查询的子查询或内查询


分类:
按子查询出现的位置:
	SELECT 后面
		仅仅支持标量子查询
	FROM 后面
		支持表子查询
	WHERE 或 HAVING 后面★
		标量子查询
		列子查询
		行子查询
	EXISTS 后面
		表子查询
按结果集的行列数不同:
	标量子查询(结果集只有一行一列)
	列子查询(结果集只有一列多行)
	行子查询(结果有一行多列)
	表子查询(结果集一般为多行多列)

还能结果集行数分类:
	单行子查询
		-只返回一行
		-使用单行比较操作符(=、>、>=、<、<=、<>)
	多行子查询
		-返回多行
		-使用多行比较操作符(IN/NOT IN、ANY|SOME、ALL)
        IN/NOT IN:等于列表中的任意一个
        ANY|SOME:和子查询返回的某一个值比较
        ALL:和子查询返回的所有值比较

示例

一、wherehaving后面

案例1:谁的工资比Abel高?
SELECT *
FROM employees
WHERE salary > (
	SELECT salary
	FROM employees
	WHERE last_name = 'Abel'
)

案例2:返回job_id与141号员工相同,salary比143号员工多的员工姓名,job_id和工资
SELECT last_name, job_id, `salary`
FROM employees
WHERE job_id = (
	SELECT job_id
	FROM employees
	WHERE `employee_id` = 141
) AND salary > (
	SELECT salary
	FROM employees
	WHERE `employee_id` = 143
);

案例3:返回公司工资最少的员工的last_name, job_id和salary
SELECT last_name, job_id, salary
FROM employees
WHERE salary = (
	SELECT MIN(salary)
	FROM employees
)

案例4:查询最低工资大于50号部门最低工资的部门id和其最低工资
SELECT `department_id`, MIN(`salary`) 最低工资
FROM employees
GROUP BY department_id
HAVING 最低工资 > (
	SELECT MIN(salary)
	FROM employees
	WHERE `department_id` = 50
);

案例5:返回location_id是14001700的部门中的所有员工姓名
SELECT last_name
FROM employees
WHERE `department_id` IN(
	SELECT DISTINCT department_id
	FROM departments
	WHERE location_id IN(1400, 1700)
);

案例6:返回其他工种比job_id为IT_PROG工种任一工资低的员工的员工号、姓名、job_id 以及salary
SELECT `employee_id`, `last_name`, `job_id`, `salary`
FROM `employees`
WHERE `salary` < ANY(
	SELECT DISTINCT salary
	FROM employees
	WHERE job_id = 'IT_PROG'
) AND job_id <> 'IT_PROG';



案例7:返回其他工种比job_id为IT_PROG工种所有工资低的员工的员工号、姓名、job_id 以及salary
SELECT `employee_id`, `last_name`, `job_id`, `salary`
FROM `employees`
WHERE `salary` < ALL(
	SELECT DISTINCT salary
	FROM employees
	WHERE job_id = 'IT_PROG'
) AND job_id <> 'IT_PROG';SELECT `employee_id`, `last_name`, `job_id`, `salary`
FROM `employees`
WHERE `salary` < (
	SELECT MIN(salary)
	FROM employees
	WHERE job_id = 'IT_PROG'
) AND job_id <> 'IT_PROG';


案例8:查询员工编号最小并且工资最高的员工信息
SELECT *
FROM employees
WHERE `employee_id` = (
	SELECT MIN(`employee_id`)
	FROM employees
) AND salary = (
	SELECT MAX(`salary`)
	FROM employees
);

可改为:
SELECT *
FROM employees
WHERE (employee_id, salary) = (
	SELECT MIN(employee_id),MAX(salary)
	FROM employees
)
------------------------------------------------------
二、select后面

案例1:查询每个部门的员工个数
SELECT d.*, (
	SELECT COUNT(*)
	FROM employees e
	WHERE e.`department_id` = d.department_id
) 个数
FROM departments d;


案例2:查询员工号=102的部门名
SELECT (
	SELECT department_name
	FROM departments d
	INNER JOIN employees e
	ON d.department_id = e.department_id
	WHERE e.`employee_id` = 102
) 部门名;

-------------------------------------------------------
三、from后面

案例:查询每个部门的平均工资的工资等级
SELECT AVG(salary),department_id
FROM employees
GROUP BY department_id


SELECT ag_dep.department_id, ag_dep.平均工资, g.`grade_level`
FROM (
	SELECT AVG(salary) 平均工资,department_id
	FROM employees
	GROUP BY department_id
	
)ag_dep
INNER JOIN job_grades g
ON ag_dep.平均工资 BETWEEN g.`lowest_sal` AND g.`highest_sal`;

----------------------------------------------------
四、exists
SELECT EXISTS(SELECT `employee_id` FROM employees);
EXISTS(A),A为null则为0,否则为1


案例1:查询有员工的部门名
SELECT department_name
FROM departments d
WHERE EXISTS(
	SELECT *
	FROM employees e
	WHERE d.`department_id` = e.`department_id`
);SELECT DISTINCT department_name
FROM departments d
WHERE department_id IN (
	SELECT department_id
	FROM employees
);	

#案例2:查询没有女朋友的男神信息
SELECT bo.*
FROM boys bo
WHERE NOT EXISTS(
	SELECT boyfriend_id
	FROM `beauty` b
	WHERE bo.`id` = b.`boyfriend_id`
);

测试

1. 查询和Zlotkey相同部门的员工姓名和工资
2. 查询工资比公司平均工资高的员工的员工号,姓名和工资。
3. 查询各部门中工资比本部门平均工资高的员工的员工号, 姓名和工资
4. 查询和姓名中包含字母u的员工在相同部门的员工的员工号和姓名
5. 查询在部门的location_id为1700的部门工作的员工的员工号
6. 查询管理者是King的员工姓名和工资
7. 查询工资最高的员工的姓名,要求first_name和last_name显示为一列,列名为 姓.名

测试题解答

1. 查询和Zlotkey相同部门的员工姓名和工资

SELECT last_name, `salary`
FROM `employees`
WHERE `department_id` = (
	SELECT department_id
	FROM employees
	WHERE last_name = 'Zlotkey'
);

2. 查询工资比公司平均工资高的员工的员工号,姓名和工资。
SELECT `employee_id`,`last_name`,`salary`
FROM employees
WHERE `salary` > (
	SELECT AVG(`salary`)
	FROM employees
);


3. 查询各部门中工资比本部门平均工资高的员工的员工号, 姓名和工资
SELECT `employee_id`,`last_name`,`salary`
FROM employees e
INNER JOIN (
	SELECT AVG(salary) ag,department_id
	FROM employees
	GROUP BY department_id
)ag_dep
ON e.department_id = ag_dep.department_id
WHERE e.salary > ag_dep.ag;

4. 查询和姓名中包含字母u的员工在相同部门的员工的员工号和姓名
SELECT `employee_id`,last_name
FROM employees
WHERE department_id IN (
	SELECT DISTINCT department_id
	FROM employees
	WHERE `last_name` LIKE '%u%'
);

5. 查询在部门的location_id为1700的部门工作的员工的员工号
SELECT `employee_id`
FROM employees
WHERE department_id IN (
	SELECT department_id
	FROM `departments`
	WHERE `location_id` = 1700
);

6. 查询管理者是K_ing的员工姓名和工资
SELECT e1.`last_name`,e1.`salary`
FROM `employees` e1
LEFT JOIN `employees` e2
ON e1.`manager_id` = e2.`employee_id`
WHERE e2.`last_name` = 'K_ing';

或者
SELECT last_name, salary
FROM employees
WHERE manager_id IN(
	SELECT `employee_id`
	FROM employees
	WHERE last_name = 'K_ing'
);

7. 查询工资最高的员工的姓名,要求first_name和last_name显示为一列,列名为 姓.SELECT CONCAT(first_name,'.',last_name) `.`
FROM employees
WHERE salary = (
	SELECT MAX(salary)
	FROM employees
);
分页查询
应用场景:当我们要显示的数据,一页显示不全,如果我们全部下载下来,但是用户只是看某几页,
这样会浪费资源,所以需要分页提交sql请求。

语法:
	SELECT 查询列表
	FROM 列表
	【join TYPE JOIN 表2】
	【on 连接条件】
	【where 筛选条件】
	【group BY 分组字段】
	【having 分组后筛选】
	【order BY 排序的字段】
	LIMIT OFFSET , size;
	
	OFFSET 要显示条目的起始索引(起始索引从0开始)
	size 要显示的条目个数

执行顺序:
①from
②join
③on
④where
⑤group BY
⑥having
⑦select
⑧order BY
⑨limit

示例

案例1:查询前五条员工信息
SELECT *
FROM employees
LIMIT 0,5;

SELECT *
FROM employees
LIMIT 5;

案例2:查询第11条到第25SELECT *
FROM employees
LIMIT 10,15;

案例3:有奖金的员工信息,并且工资较高的前10名显示出来
SELECT *
FROM employees
WHERE commission_pct IS NOT NULL
ORDER BY salary DESC
LIMIT 10;
union联合查询
UNION 联合,合并:将多条查询语句的结果合并成一个结果
语法:
查询语句1
UNION
查询语句2
....

应用场景:
要查询的结果来源于多个表,且多个表没有直接的连接关系,但查询的信息一致时

特点:
1、要求多条查询语句的查询列数是一致的
2、要求多条查询语句的查询的每一列的类型和顺序最好一致
3union关键字默认去重,如果使用union all可以包含重复项

引入的案例:查询部门编号>90或邮箱包含a的员工信息
SELECT *
FROM employees
WHERE email LIKE '%a%' OR department_id > 90;

使用union
SELECT * FROM employees WHERE email LIKE '%a%'
UNION SELECT * FROM employees WHERE department_id > 90;

案例:查询中国用户中男性的信息以及外国用户中男性用户的信息
SELECT id,cname,csex FROM t_ca WHERE csex = '男'
UNION
SELECT t_id,tName,tGender FROM t_ua WHERE tGender='male';

DML语言

数据操作语言:
插入:insert
修改:update
删除:delete
插入语句

方式一:经典的插入

/*
语法:
insert into 表名(列名,...) values(值1,...);

*/
SELECT * FROM beauty;

#1.插入的值的类型要与列的类型一致或兼容
INSERT INTO beauty(id,NAME,sex,borndate,phone,photo,boyfriend_id)
VALUES(13,'唐艺昕','女','1990-4-23','1898888888',NULL,2);

#2.不可以为null的列必须插入值。可以为null的列如何插入值?
#方式一:
INSERT INTO beauty(id,NAME,sex,borndate,phone,photo,boyfriend_id)
VALUES(13,'唐艺昕','女','1990-4-23','1898888888',NULL,2);

#方式二:
INSERT INTO beauty(id,NAME,sex,phone)
VALUES(15,'娜扎','女','1388888888');


#3.列的顺序是否可以调换
INSERT INTO beauty(NAME,sex,id,phone)
VALUES('蒋欣','女',16,'110');


#4.列数和值的个数必须一致

INSERT INTO beauty(NAME,sex,id,phone)
VALUES('关晓彤','女',17,'110');

#5.可以省略列名,默认所有列,而且列的顺序和表中列的顺序一致

INSERT INTO beauty
VALUES(18,'张飞','男',NULL,'119',NULL,NULL);

方式二

/*

语法:
insert into 表名
set 列名=值,列名=值,...
*/


INSERT INTO beauty
SET id=19,NAME='刘涛',phone='999';

两种方式比较

#1、方式一支持插入多行,方式二不支持

INSERT INTO beauty
VALUES(23,'唐艺昕1','女','1990-4-23','1898888888',NULL,2)
,(24,'唐艺昕2','女','1990-4-23','1898888888',NULL,2)
,(25,'唐艺昕3','女','1990-4-23','1898888888',NULL,2);

#2、方式一支持子查询,方式二不支持

INSERT INTO beauty(id,NAME,phone)
SELECT 26,'宋茜','11809866';

INSERT INTO beauty(id,NAME,phone)
SELECT id,boyname,'1234567'
FROM boys WHERE id<3;
修改语句
/*
1.修改单表的记录★

语法:
update 表名
set 列=新值,列=新值,...
where 筛选条件;

2.修改多表的记录【补充】

语法:
sql92语法:
update 表1 别名,表2 别名
set 列=值,...
where 连接条件
and 筛选条件;

sql99语法:
update 表1 别名
inner|left|right join 表2 别名
on 连接条件
set 列=值,...
where 筛选条件;

*/

#1.修改单表的记录
#案例1:修改beauty表中姓唐的女神的电话为13899888899

UPDATE beauty SET phone = '13899888899'
WHERE NAME LIKE '唐%';

#案例2:修改boys表中id好为2的名称为张飞,魅力值 10
UPDATE boys SET boyname='张飞',usercp=10
WHERE id=2;

#2.修改多表的记录

#案例 1:修改张无忌的女朋友的手机号为114
UPDATE boys bo
INNER JOIN beauty b ON bo.`id`=b.`boyfriend_id`
SET b.`phone`='119',bo.`userCP`=1000
WHERE bo.`boyName`='张无忌';

#案例2:修改没有男朋友的女神的男朋友编号都为2号

UPDATE boys bo
RIGHT JOIN beauty b ON bo.`id`=b.`boyfriend_id`
SET b.`boyfriend_id`=2
WHERE bo.`id` IS NULL;

SELECT * FROM boys;
删除语句
/*
方式一:delete
语法:

1、单表的删除【★】
delete from 表名 where 筛选条件

2、多表的删除【补充】

sql92语法:
delete 表1的别名,表2的别名
from 表1 别名,表2 别名
where 连接条件
and 筛选条件;

sql99语法:

delete 表1的别名,表2的别名
from 表1 别名
inner|left|right join 表2 别名 on 连接条件
where 筛选条件;



方式二:truncate
语法:truncate table 表名;

*/


方式一:delete

#1.单表的删除
#案例:删除手机号以9结尾的女神信息

DELETE FROM beauty WHERE phone LIKE '%9';


#2.多表的删除

#案例:删除张无忌的女朋友的信息

DELETE b
FROM beauty b
INNER JOIN boys bo ON b.`boyfriend_id` = bo.`id`
WHERE bo.`boyName`='张无忌';


#案例:删除黄晓明的信息以及他女朋友的信息
DELETE b,bo
FROM beauty b
INNER JOIN boys bo ON b.`boyfriend_id`=bo.`id`
WHERE bo.`boyName`='黄晓明';

方式二:truncate

#案例:将魅力值>100的男神信息删除
TRUNCATE TABLE boys ;

delete vs truncate

1.delete 可以加where 条件,truncate不能加

2.truncate删除,效率高一丢丢

3.假如要删除的表中有自增长列,
如果用delete删除后,再插入数据,自增长列的值从断点开始,
而truncate删除后,再插入数据,自增长列的值从1开始。

4.truncate删除没有返回值,delete删除有返回值

5.truncate删除不能回滚,delete删除可以回滚.

DDL语言

数据定义语言:库和表的管理
创建: CREATE
修改: ALTER
删除: DROP

库和表的管理
一、库的管理
1.库的创建
语法:
CREATE DATABASE [IF NOT EXISTS] 库名;

2.库的修改
更改库的字符集
ALTER DATABASE 库名 CHARACTER SET gbk/utf-8;

3.库的删除
DROP DATABASE IF EXISTS 库名;
-------------------------------------------------
二、表的管理
1.表的创建
CREATE TABLE 表名(
	列名 列名的类型【长度 约束】,
	列名 列名的类型【长度 约束】,
	....
	列名 列名的类型【长度 约束】,
);

#案例:创建表Book
CREATE TABLE IF EXISTS book(
	id INT,
	bName VARCHAR(20),
	price DOUBLE,
	author VARCHAR(20),
	publishDate DATETIME
);

DESC book;

2.表的修改
①修改列名
ALTER TABLE book CHANGE [COLUMN] publishDate pubDate DATETIME;

②修改列的类型或约束
ALTER TABLE book MODIFY COLUMN pubdate TIMESTAMP;

③添加列
ALTER TABLE book ADD COLUMN annual DOUBLE;

④删除列
ALTER TABLE book DROP COLUMN annual;

⑤修改表名
ALTER TABLE book RENAME TO books;

3.表的删除
DROP TABLE IS EXISTS book;

4.表的复制
①仅仅复制表的结构
CREATE TABLE copy_author LIKE author;

②复制表的结构+数据
CREATE TABLE copy_author
SELECT * FROM author;

练习

#1.	创建表dept1
NAME	NULL?	TYPE
id		INT(7)
NAME		VARCHAR(25)


USE test;

CREATE TABLE dept1(
	id INT(7),
	NAME VARCHAR(25)
	

);
#2.	将表departments中的数据插入新表dept2中

CREATE TABLE dept2
SELECT department_id,department_name
FROM myemployees.departments;


#3.	创建表emp5
NAME	NULL?	TYPE
id		INT(7)
First_name	VARCHAR (25)
Last_name	VARCHAR(25)
Dept_id		INT(7)

CREATE TABLE emp5(
id INT(7),
first_name VARCHAR(25),
last_name VARCHAR(25),
dept_id INT(7)

);

#4.	将列Last_name的长度增加到50

ALTER TABLE emp5 MODIFY COLUMN last_name VARCHAR(50);
#5.	根据表employees创建employees2

CREATE TABLE employees2 LIKE myemployees.employees;

#6.	删除表emp5
DROP TABLE IF EXISTS emp5;

#7.	将表employees2重命名为emp5

ALTER TABLE employees2 RENAME TO emp5;

#8.在表dept和emp5中添加新列test_column,并检查所作的操作

ALTER TABLE emp5 ADD COLUMN test_column INT;
#9.直接删除表emp5中的列 dept_id
DESC emp5;
ALTER TABLE emp5 DROP COLUMN test_column;
常见数据类型

一、数值型
1、整型
tinyint、smallint、mediumint、int/integer、bigint
1 2 3 4 8 字节

特点:
①都可以设置无符号和有符号,默认有符号,通过unsigned设置无符号
②如果超出了范围,会报out or range异常,插入临界值
③长度可以不指定,默认会有一个长度
长度代表显示的最大宽度,如果不够则左边用0填充,但需要搭配zerofill,并且默认变为无符号整型

2、浮点型
定点数:decimal(M,D)/dec(M,D)
浮点数:
float(M,D) 4
double(M,D) 8

特点:
①M代表整数部位+小数部位的个数,D代表小数部位
②如果超出范围,则报out or range异常,并且插入临界值
③M和D都可以省略,但对于定点数,M默认为10,D默认为0
④如果精度要求较高,则优先考虑使用定点数

二、字符型
char、varchar、binary、varbinary、enum、set、text、blob

char:固定长度的字符,写法为char(M),最大长度不能超过M,其中M可以省略,默认为1
varchar:可变长度的字符,写法为varchar(M),最大长度不能超过M,其中M不可以省略

三、日期型
year年
date日期
time时间
datetime 日期+时间 8
timestamp 日期+时间 4 比较容易受时区、语法模式、版本的影响,更能反映当前时区的真实时间

datetime 和timestamp比较:

​ 字节 范围 时区等的影响
datetime 8 1000-9999 不受
timestamp 4 1970-2038 受

常见约束

含义:一种限制,用于限制表中的数据,为了保证数据的准确性和可靠性
CREATE TABLE 表名(
字段名 字段类型 约束
字段名 字段类型 约束

​ …
​ 表级约束

)

一、常见的约束

NOT NULL:非空,该字段的值必填
UNIQUE:唯一,该字段的值不可重复,可以为空
DEFAULT:默认,该字段的值不用手动插入有默认值
CHECK:检查,mysql不支持
PRIMARY KEY:主键,该字段的值不可重复并且非空 unique+not null
FOREIGN KEY:外键,该字段的值引用了另外的表的字段

主键和唯一★
1、区别:
①、一个表至多有一个主键,但可以有多个唯一
②、主键不允许为空,唯一可以为空(虽然唯一可以为空,但是只允许一个为空)
2、相同点
都具有唯一性
都支持组合键,但不推荐

外键:
1、用于限制两个表的关系,从表的字段值引用了主表的某字段值
2、外键列和主表的被引用列要求类型一致或兼容,意义一样,名称无要求
3、主表的被引用列要求是一个key(一般就是主键)
4、插入数据,先插入主表
删除数据,先删除从表
可以通过以下两种方式来删除主表的记录
#方式一:级联删除
ALTER TABLE stuinfo ADD CONSTRAINT fk_stu_major FOREIGN KEY(majorid) REFERENCES major(id) ON DELETE CASCADE;

#方式二:级联置空
ALTER TABLE stuinfo ADD CONSTRAINT fk_stu_major FOREIGN KEY(majorid) REFERENCES major(id) ON DELETE SET NULL;

约束的添加分类:
列级约束:六大约束语法都支持,但外键约束没有效果,不可以起约束名
表级约束:除了非空、默认,其它都支持,可以起约束名但主键没有效果

二、创建表时添加约束

1.添加列级约束

CREATE TABLE stuinfo(
	id INT PRIMARY KEY,
	stuName VARCHAR(20) NOT NULL,
	gender CHAR(1) CHECK(gender='男' OR gender='女'),
	seat INT UNIQUE,
	age INT DEFAULT 18,
	majorId INT FOREIGN KEY REFERENCES major(id) #外键
);

CREATE TABLE major(
	id INT PRIMARY KEY,
	majorName VARCHAR(20)
);


2.添加表级约束
DROP TABLE IF EXISTS stuinfo;
CREATE TABLE stuinfo(
	id INT,
	stuName VARCHAR(20),
	gender CHAR(1),
	seat INT,
	age INT,
	majorId INT,
	
	CONSTRAINT pk PRIMARY KEY(id),
	CONSTRAINT uq UNIQUE(seat),
	CONSTRAINT ck CHECK(gender = '男' OR gender = '女')
	CONSTRAINT fk_stuinfo_major FOREIGN KEY(majorId) REFERENCES major(id)
);


通用的写法:
CREATE TABLE IF EXISTS stuinfo(
	id INT PRIMARY KEY,
	stuName VARCHAR(20) NOT NULL,
	gender CHAR(1),
	age INT DEFAULT 18,
	majorid INT,
	CONSTRAINT fk_stuinfo_major FOREIGN KEY(marjorid) REFERENCES major(id)
)

三、修改表时添加或删除约束

/*
列级约束:alter table 表名 modify column 字段名 字段类型 约束名;
表级约束:alter table 表名 add【 constraint 约束名】 primary key(字段名);
*/

1、非空
添加非空
alter table 表名 modify column 字段名 字段类型 not null;
删除非空
alter table 表名 modify column 字段名 字段类型 ;

2、默认
添加默认
alter table 表名 modify column 字段名 字段类型 default;
删除默认
alter table 表名 modify column 字段名 字段类型 ;

3、主键
添加主键
alter table 表名 modify column 字段名 字段类型 primary key;
alter table 表名 addconstraint 约束名】 primary key(字段名);
删除主键
alter table 表名 drop primary key;

4、唯一
添加唯一
alter table 表名 addconstraint 约束名】 unique(字段名);
删除唯一
alter table 表名 drop index 索引名;

5、外键
添加外键
alter table 表名 addconstraint 约束名】 foreign key(字段名) references 主表(被引用列);
删除外键
alter table 表名 drop foreign key 约束名;


测试:
1. 向表emp2的id列中添加PRIMARY KEY约束(my_emp_id_pk)
2. 向表dept2的id列中添加PRIMARY KEY约束(my_dept_id_pk)
3. 向表emp2中添加列dept_id,并在其中定义FOREIGN KEY约束,与之相关联的列是dept2表中的id列。

1.
ALTER TABLE emp2 ADD CONSTRAINT my_emp_id_pk PRIMARY KEY(id);

2.
ALTER TABLE dept2 ADD CONSTRAINT my_dept_id_pk PRIMARY KEY(id);

3.
ALTER TABLE ADD COLUMN dept_id INT;
ALTER TABLE ADD CONSTRAINT fk_emp2_dept2 FOREIGN KEY(dept_id) REFERENCES dept2(id);

四、自增长列

又称标识列

特点:
1、不用手动插入值,可以自动提供序列值,默认从1开始,步长为1
auto_increment_increment
如果要更改起始值:手动插入值
如果要更改步长:更改系统变量
show variables like '%auto_increment%'
set auto_increment_increment=值;
2、一个表至多有一个自增长列
3、自增长列只能支持数值型
4、自增长列必须为一个key

一、创建表时设置自增长列
create table 表(
	字段名 字段类型 约束 auto_increment
)

二、修改表时设置自增长列
alter table 表 modify column 字段名 字段类型 约束 auto_increment

三、删除自增长列
alter table 表 modify column 字段名 字段类型 约束 

TCL语言

Transaction Contrl Language 事务控制语言

一、含义

事务:一个或一组sql语句组成一个执行单元,这个执行单元要么全部执行,要么全部不执行。

二、特点(ACID)

A 原子性:一个事务是不可再分割的整体,要么都执行要么都不执行
C 一致性:一个事务可以使数据从一个一致状态切换到另外一个一致的状态
I 隔离性:一个事务不受其他事务的干扰,多个事务互相隔离的
D 持久性:一个事务一旦提交了,则永久的持久化到本地

三、事务的使用步骤

  • 隐式(自动)事务:没有明显的开启和结束,本身就是一条事务可以自动提交,比如insert、update、delete系统本身有设置自动提交功能
    SHOW VARIABLES LIKE ‘autocommit’; #默认为ON,打开状态

  • 显式事务:具有明显的开启和结束
    前提必须先关闭自动提交功能
    SET autocommit = 0; #OFF

步骤一:开启事务

set autocommit=0;
start transaction;#可以省略

步骤二、编写事务中的sql语句(select、insert、update、delete)

语句1
语句2

步骤三:结束事务

提交:commit;
回滚:rollback;
回滚到指定的地方:rollback to 回滚点名;

四、并发事务

1、事务的并发问题是如何发生的?

​ 多个事务 同时操作同一个数据库的相同数据时

2、并发问题都有哪些?

  • 脏读:一个事务读取了其他事务还没有提交的数据,读到的是其他事务“更新”的数据
  • 不可重复读:一个事务多次读取,结果不一样
  • 幻读:一个事务读取了其他事务还没有提交的数据,只是读到的是其他事务“插入”的数据

3、如何解决并发问题

​ 通过设置隔离级别来解决并发问题

4、隔离级别
脏读 不可重复读 幻读
read uncommitted: 读未提交 × × ×
read committed:读已提交 √ × ×
repeatable read:可重复读 √ √ ×
serializable:串行化 √ √ √

mysql默认隔离级别:repeatable read

oracle中默认:read committed

1.查看隔离级别
SELECT @@tx_isolation;

2.修改隔离级别
设置当前mysql连接的隔离级别:
SET SESSION TRANSACTION ISOLATION LEVEL READ UNCOMMITTED;
设置数据库系统的全局的隔离级别:
SET GLOBAL SESSION TRANSACTION ISOLATION LEVEL READ UNCOMMITTED;

delete 和 truncate在事务使用时的区别

#delete
SET autocmmit = 0;
START TRANSACTION;
DELETE FROM account;
ROLLBACK;

#truncate
SET autocmmit = 0;
START TRANSACTION;
TRUNCATE TABLE account;
ROLLBACK;

delete支持回滚,truncate不支持回滚

四、其它

视图

含义:虚拟表,和普通表一样使用。它的数据来自于表,通过执行时动态生成

视图的好处:

  • 简化了sql语句,不必知道查询细节
  • 重用sql语句
  • 保护数据,提高安全性

一、视图的创建

1.查询邮箱中包含a字符的员工名、部门名和工种信息
①创建视图
CREATE VIEW myv1
AS
SELECT last_name, department_name,job_title
FROM employees e
JOIN departments d ON e.department_id = d.department_id
JOIN jobs j ON j.job_id = e.job_id;

②使用视图

SELECT * FROM myv1 WHERE last_name LIKE '%a%';


2.查询各部门的平均工资级别
①创建视图
查询每个部门的平均工资
CREATE VIEW myv2
AS
SELECT AVG(salary) ag,department_id
FROM employees
GROUP BY department_id;

②使用视图,查询平均工资级别
SELECT myv2.`ag`,g.`grade_level` 
FROM myv2
JOIN `job_grades` g
ON myv2.`ag` BETWEEN g.`lowest_sal` AND g.`highest_sal`;


3.查询平均工资最低的部门信息
SELECT * FROM myv2 ORDER BY ag LIMIT 1;

4.查询平均工资最低的部门名和工资

CREATE VIEW myv3
AS
SELECT * FROM myv2 ORDER BY ag LIMIT 1;

SELECT d.`department_name`,m.ag
FROM myv3 m
JOIN departments d
ON m.department_id = d.`department_id`;

二、视图的修改

方式一:
/*
CREATE OR REPLACE VIEW 视图名
AS
查询语句;
*/

SELECT * FROM myv3
CREATE OR REPLACE VIEW myv3
AS
SELECT AVG(salary),job_id
FROM employees
GROUP BY job_id;

方式二:
/*
ALTER VIEW 视图名
AS
查询语句;
*/

ALTER VIEW myv3
AS
SELECT * FROM employees;

三、删除视图

/*
语法:
DROP VIEW 视图名,视图名...
*/

DROP VIEW myv3;

四、查看视图

DESC 视图名;  #显示结构
SHOW CREATE VIEW 视图名;  #显示创建语句过程。

测试题

一、创建视图emp_v1,要求查询电话号码以‘011’开头的员工姓名和工资、邮箱
CREATE VIEW emp_v1
AS
SELECT last_name,salary,`email`
FROM employees
WHERE `phone_number` LIKE '011%';

二、要求将视图emp_v1修改为查询电话号码以‘011’开头的并且邮箱中包含e字符的员工姓名和工资、邮箱
CREATE OR REPLACE VIEW emp_v1
AS
SELECT last_name,salary,`email`
FROM employees
WHERE `phone_number` LIKE '011%'
AND `email` LIKE '%e%';

三、创建视图emp_v2,要求查询部门的最高工资高于12000的部门信息
CREATE OR REPLACE VIEW emp_v2
AS
SELECT d1.*
FROM `departments` d1
INNER JOIN (
	SELECT MAX(`salary`) 最高工资,department_id
	FROM `employees`
	GROUP BY department_id
	HAVING 最高工资 > 12000
) d2
ON d1.department_id = d2.department_id;

注意:低版本的mysql,视图是不支持子查询的

可以分两步:
①创建中间视图
CREATE OR REPLACE VIEW emp_v2
AS
SELECT MAX(`salary`) 最高工资,department_id
FROM `employees`
GROUP BY department_id
HAVING 最高工资 > 12000;

②使用视图
SELECT d1.*
FROM `departments` d1
INNER JOIN emp_v2 d2
ON d1.department_id = d2.department_id;

五、视图的更新

CREATE OR REPLACE VIEW myv1
AS
SELECT last_name,email
FROM employees;

1.插入
INSERT INTO myv1 VALUES('张飞','zf@qq.com');

SELECT *
FROM myv1;

注意:视图更新了,原始表也会更新。

2.修改
UPDATE myv1 SET last_name = '张无忌' WHERE last_name = '张飞';

注意:视图更新了,原始表也会更新。

3.删除
DELETE FROM myv1 WHERE last_name = '张无忌';

注意:视图更新了,原始表也会更新。

***********************************************************
注意:视图一般用于查询的,而不是更新的,所以具备以下特点的视图都不允许更新
①包含分组函数、group bydistincthavingunion、
②join
③常量视图
④where后的子查询用到了from中的表
⑤用到了不可更新的视图

视图和表的对比

		 关键字		 是否占用物理空间			   使用
视图	    view		占用较小,只保存sql逻辑		一般用于查询
表		table		   保存实际的数据			    增删改查

变量

系统变量:

  • 全局变量
  • 会话变量

自定义变量:

  • 用户变量
  • 局部变量

一、系统变量

说明:变量由系统提供,不是用户自定义
使用的语法:
1.查看所有的系统变量
show global variables; #全局
show session variables; #会话

2.查看满足条件的部分系统变量
show global|[session] variables like '%xxx'

3.查看指定的某个系统变量的值
SELECT @@系统变量名;
SELECT @@global|session.系统变量名

4.为某个系统变量赋值
方式一:
set global|session 系统变量名 =;

方式二:
set @@global|[session].系统变量名=;


(一)全局变量
作用域:服务器每次启动将为所有的全局变量赋初始值,针对于所有的会话有效,但是不能跨重启。

(二)会话变量
作用域:仅仅针对当前会话有效

二、自定义变量

  • 用户变量
说明:用户自己定义,不是系统的
使用步骤:
①声明
②赋值
③使用(查看、比较、运算等)

作用域:针对当前会话(连接)有效,同于会话变量的作用域,应用在任何地方。

step1.声明并初始化
set @用户变量名 =;
set @用户变量名 :=;
select @用户变量名 :=;

step2.赋值(更新用户变量的值)
方式一:通过setselect
    set @用户变量名 =;
    set @用户变量名 :=;
    select @用户变量名 :=;
方式二:通过select into
	select 字段 into @用户变量名
	from;

step3.使用
select @用户变量名;

  • 局部变量
作用域:仅仅在定义它的begin end中有效

step1.声明
DECLARE 变量名 类型;
DECLARE 变量名 default 类型;

step2.赋值
方式一:通过set或select
    set 局部变量名 = 值;
    set 局部变量名 := 值;
    select 局部变量名 := 值;
方式二:通过select into
	select 字段 into 局部变量名
	from 表;
	
step3.使用
select 局部变量名;
  • 用户变量 vs 局部变量
             作用域             定义和使用的位置
用户变量      当前会话            会话的任何地方
局部变量      begin-end中        只能在begin-end中,且为第一句话

存储过程和函数

好处:
1、提高代码的复用性
2、简化操作
3、减少编译次数并且减少了服务器的连接次数,提高了效率

一组预先编译好的sql语句的集合,理解成批处理语句

存储过程

一、创建

/*
语法:
DELIMITER 结束标记
CREATE PROCEDURE 存储过程名(参数列表)
BEGIN
	存储过程体(一组sql语句)
END 结束标记
*/
注意:
1.参数列表包含三部分
参数模式   参数名   参数类型
举例:
IN stuname VARCHAR(20)

参数模式:
IN: 该参数可以作为输入,需要传入值
OUT: 该参数可以作为输出,也就是可以作为返回值
INOUT:该参数既可以作为输入也可以作为输出

2.如果存储过程体仅仅只有一句话,begin END 可以省略
存储过程体的每条SQL语句的结尾要求必须加上分号。
存储过程的结尾可以使用 delimter 重新设置

二、调用

语法:
CALL 存储过程名(实参列表);

1.空参列表

案例:插入到admin表中五条记录

SELECT * FROM admin;
DELIMITER $
CREATE PROCEDURE myp1()
BEGIN
	INSERT INTO admin(username,`password`)
	VALUES('john1','0000'),('lily','0000'),('rose','0000'),('tom','0000'),('jack','0000');
END $

调用
CALL myp1()

2.创建带in模式参数的存储过程

案例1:创建存储过程实现根据女神名,查询对应的男神信息
DELIMITER $
CREATE PROCEDURE myp2(IN beautyName VARCHAR(20))
BEGIN
	SELECT bo.*
	FROM boys bo
	RIGHT JOIN `beauty` b ON bo.id = b.`boyfriend_id`
	WHERE b.name = beautyName;
END $

SELECT * FROM `beauty`;

调用
CALL myp2('热巴')
案例2:创建存储过程,判断用户是否存在
DELIMITER $
CREATE PROCEDURE myp3(IN username VARCHAR(20), IN `password` VARCHAR(20))
BEGIN
	DECLARE result INT DEFAULT 0;
	
	SELECT COUNT(*) INTO result
	FROM admin a
	WHERE a.username = username
	AND a.`password` = `password`;
	
	SELECT IF(result>0,'成功','失败') result;
END $

调用
CALL myp3('张飞','0000')

3.创建带out模式的存储过程

案例1:根据女神名,返回对应的男神名

DELIMITER $
CREATE PROCEDURE myp4(IN beautyName VARCHAR(20), OUT boyName VARCHAR(20))
BEGIN
	SELECT bo.`boyName` INTO boyName
	FROM boys bo
	INNER JOIN beauty b ON bo.id = b.boyfriend_id
	WHERE b.name = beautyName;
END $

SET @bName$
CALL myp4('小昭',@bName)

SELECT @bName;

案例2:根据女神名,返回对应的男神名和男神魅力值
DELIMITER $
CREATE PROCEDURE myp5(IN beautyName VARCHAR(20), OUT boyName VARCHAR(20), OUT userCP INT)
BEGIN
	SELECT bo.boyName, bo.`userCP` INTO boyName, userCP
	FROM boys bo
	INNER JOIN beauty b ON bo.id = b.boyfriend_id
	WHERE b.name = beautyName;
END $

CALL myp5('小昭',@bName,@userCP)

SELECT @bName, @userCP;

4.创建带inout模式参数的存储过程

案例1:传入a和b两个值,最终a和b都翻倍并返回
DELIMITER $
CREATE PROCEDURE myp6(INOUT a INT, INOUT b INT)
BEGIN
	SET a = a*2;
	SET b = b*2;
END $

SET @m=10;
SET @n=20;
CALL myp6(@m,@n);

SELECT @m,@n;

测试

1、创建存储过程或函数实现传入用户名和密码,插入到admin表中
DELIMITER $
CREATE PROCEDURE myp1(IN userName VARCHAR(10), IN `password` VARCHAR(10))
BEGIN
	INSERT INTO admin(`username`,`password`)
	VALUES(userName,`password`);
END $

CALL myp1('jack','1000');
SELECT * FROM `admin`;

2、创建存储过程或函数实现传入女神编号,返回女神名称和女神电话
DELIMITER $
CREATE PROCEDURE myp2(IN beautyId INT, OUT beautyName VARCHAR(20), OUT phone VARCHAR(11))
BEGIN
	SELECT `name`,b.`phone` INTO beautyName, phone
	FROM `beauty` b
	WHERE `id` = beautyId;
END $

SET @beautyName;
SET @phone;
CALL myp2('2',@beautyName,@phone);

SELECT @beautyName,@phone;
SELECT * FROM `beauty`;


3、创建存储存储过程或函数实现传入两个女神生日,返回大小
DELIMITER $
CREATE PROCEDURE myp3(IN birth1 DATETIME, IN birth2 DATETIME, OUT result INT)
BEGIN
	SELECT DATEDIFF(birth1,birth2) INTO result;
END $

4、创建存储过程或函数实现传入一个日期,格式化成xx年xx月xx日并返回
DELIMITER $
CREATE PROCEDURE myp4(IN date_time DATETIME, OUT new_date)
BEGIN
	SELECT DATE_FORMAT(date_time,'%y年%m月%d日') INTO new_date;
END $

5、创建存储过程或函数实现传入女神名称,返回:女神 AND 男神 格式的字符串
如 传入 :小昭
返回: 小昭 AND 张无忌

DELIMITER $
CREATE PROCEDURE myp5(IN beautyName VARCHAR(20), OUT strBeauty_AND_Boy VARCHAR(20))
BEGIN
	SELECT CONCAT(IFNULL(beautyName,'null'),' AND ',bo.`boyName`) INTO strBeauty_AND_Boy
	FROM `boys` bo
	INNER JOIN `beauty` b
	ON bo.id = b.`boyfriend_id`
	WHERE b.`name` = beautyName;
END $

6、创建存储过程或函数,根据传入的条目数和起始索引,查询beauty表的记录
DELIMITER $
CREATE PROCEDURE myp6(IN startIndex INT, IN size INT)
BEGIN
	SELECT * FROM beauty LIMIT startIndex,size;
END $

三、删除

语法:drop procedure 存储过程名;

四、查看

SHOW CREATE PROCEDURE 存储过程名;

函数

函数与存储过程的区别

存储过程:可以有0个返回也可以有多个返回,适合批量的插入、批量更新
函数:有且仅有一个返回,适合做处理数据后返回一个结果

一、创建

CREATE FUNCTION 函数名(参数列表) RETURNS 返回类型
BEGIN
	函数体
END

1.参数列表 包含两部分:
参数名  参数类型

2.函数体:
肯定会有return 语句,如果没有会报错。
如果return没有放到最后也不报错,但不建议

3.如果函数体只有一条语句,可省略begin END
4.使用delimiter语句设置结束标记

二、调用

语法:

SELECT 函数名(参数列表)

1.无参有返回

案例:返回公司的员工数
DELIMITER $
CREATE FUNCTION myf1() RETURNS INT
BEGIN
	DECLARE c INT DEFAULT 0;
	SELECT COUNT(*) INTO c
	FROM employees;
	RETURN c;
END $


SELECT myf1();

2.有参有返回

案例1:根据员工名返回他的工资
DELIMITER $
CREATE FUNCTION myf2(empName VARCHAR(20)) RETURNS DOUBLE
BEGIN
	SET @sal = 0; #定义用户变量
	SELECT salary
	FROM employees
	WHERE last_name = empName;
	
	RETURN @sal;
END $

SELECT * FROM employees

SELECT myf2('Kochhar')

案例2:根据部门名返回该部门的平均工资
DELIMITER $
CREATE FUNCTION myf3(deptName VARCHAR(20)) RETURNS DOUBLE
BEGIN
	DECLARE sal DOUBLE;
	SELECT AVG(salary) INTO sal
	FROM employees e
	JOIN departments d ON e.department_id = d.department_id
	WHERE d.department_name = deptName;
	
	RETURN sal;
END $

三、查看函数

SHOW CREATE FUNCTION 函数名;

四、删除函数

DROP FUNCTION 函数名;

测试

1、创建函数,实现传入两个float,返回二者之和
DELIMITER $
CREATE FUNCTION funSUM(num1 FLOAT, num2 FLOAT) RETURNS FLOAT
BEGIN
	DECLARE mysum FLOAT DEFAULT 0;
	SET mysum = num1 + num2;
	RETURN mysum;
END $

SELECT funSUM(1,2);

2、创建函数,实现传入工种名,返回该工种的员工人数
3、创建函数,实现传入员工名,返回该员工的领导名

流程控制结构

  • 顺序结构:程序从上往下依次执行
  • 分支结构:程序可以从两条或多条路径中选择一条去执行
  • 循环结构:程序满足一定条件的基础上,重复执行一段代码
分支结构

1.if函数

功能:实现简单的双分支
语法:
IF(表达式1,表达式2,表达式3)
执行顺序:
如果表达式1成立,则返回表达式2的值,否则返回表达式3的值

应用:任何地方

2.case结构

情况1:类似于java中的switch语句,一般用于实现等值判断
语法:
	CASE 变量|表达式|字段
	WHEN 要判断的值 THEN 返回的值1或语句1
	WHEN 要判断的值 THEN 返回的值2或语句2
	...
	ELSE 要返回的值n或语句n
	END CASE;
	
情况2:类似于java中的多重IF语句,一般用于区间判断
语法:
	CASE 
	WHEN 要判断的条件1 THEN 返回的值1或语句1
	WHEN 要判断的条件2 THEN 返回的值2或语句2
	...
	ELSE 要返回的值n或语句n
	END CASE;

特点:
①
可以放在任何位置,
如果放在begin end 外面,作为表达式结合着其他语句使用
如果放在begin end 里面,一般作为独立的语句使用

②
如果某个when后面的条件成立,则执行对应then后面的语句,并且结束case
如果都不满足则执行ELSE后语句

③ELSE可以省略,如果省略且所有的WHEN条件也都不满足则返回NULL

3.if结构

功能:实现多重分支
/*
语法:
if 条件1 then 语句1;
elseif 条件2 then 语句2;
....
else 语句n;
end if;
功能:类似于多重if

只能应用在begin end 中

*/

案例

#案例1:创建函数,实现传入成绩,如果成绩>90,返回A,如果成绩>80,返回B,如果成绩>60,返回C,否则返回D

CREATE FUNCTION test_if(score FLOAT) RETURNS CHAR
BEGIN
	DECLARE ch CHAR DEFAULT 'A';
	IF score>90 THEN SET ch='A';
	ELSEIF score>80 THEN SET ch='B';
	ELSEIF score>60 THEN SET ch='C';
	ELSE SET ch='D';
	END IF;
	RETURN ch;
	
	
END $

SELECT test_if(87)$

#案例2:创建存储过程,如果工资<2000,则删除,如果5000>工资>2000,则涨工资1000,否则涨工资500


CREATE PROCEDURE test_if_pro(IN sal DOUBLE)
BEGIN
	IF sal<2000 THEN DELETE FROM employees WHERE employees.salary=sal;
	ELSEIF sal>=2000 AND sal<5000 THEN UPDATE employees SET salary=salary+1000 WHERE employees.`salary`=sal;
	ELSE UPDATE employees SET salary=salary+500 WHERE employees.`salary`=sal;
	END IF;
	
END $

CALL test_if_pro(2100)$

#案例3:创建函数,实现传入成绩,如果成绩>90,返回A,如果成绩>80,返回B,如果成绩>60,返回C,否则返回D

CREATE FUNCTION test_case(score FLOAT) RETURNS CHAR
BEGIN 
	DECLARE ch CHAR DEFAULT 'A';
	
	CASE 
	WHEN score>90 THEN SET ch='A';
	WHEN score>80 THEN SET ch='B';
	WHEN score>60 THEN SET ch='C';
	ELSE SET ch='D';
	END CASE;
	
	RETURN ch;
END $

SELECT test_case(56)$

循环结构

分类:
while、loop、repeat

循环控制:

iterate类似于 continue,继续,结束本次循环,继续下一次
leave 类似于 break,跳出,结束当前所在的循环

1.while

语法:

【标签:】while 循环条件 do
	循环体;
end while【 标签】;

联想:

while(循环条件){

	循环体;
}

2.loop

语法:
【标签:】loop
	循环体;
end loop 【标签】;

可以用来模拟简单的死循环



3.repeat

语法:
【标签:】repeat
	循环体;
until 结束循环的条件
end repeat 【标签】;

三者对比

案例

#1.没有添加循环控制语句
#案例:批量插入,根据次数插入到admin表中多条记录
DROP PROCEDURE pro_while1$
CREATE PROCEDURE pro_while1(IN insertCount INT)
BEGIN
	DECLARE i INT DEFAULT 1;
	WHILE i<=insertCount DO
		INSERT INTO admin(username,`password`) VALUES(CONCAT('Rose',i),'666');
		SET i=i+1;
	END WHILE;
	
END $

CALL pro_while1(100)$


/*

int i=1;
while(i<=insertcount){

	//插入
	
	i++;

}

*/


#2.添加leave语句

#案例:批量插入,根据次数插入到admin表中多条记录,如果次数>20则停止
TRUNCATE TABLE admin$
DROP PROCEDURE test_while1$
CREATE PROCEDURE test_while1(IN insertCount INT)
BEGIN
	DECLARE i INT DEFAULT 1;
	a:WHILE i<=insertCount DO
		INSERT INTO admin(username,`password`) VALUES(CONCAT('xiaohua',i),'0000');
		IF i>=20 THEN LEAVE a;
		END IF;
		SET i=i+1;
	END WHILE a;
END $


CALL test_while1(100)$


#3.添加iterate语句

#案例:批量插入,根据次数插入到admin表中多条记录,只插入偶数次
TRUNCATE TABLE admin$
DROP PROCEDURE test_while1$
CREATE PROCEDURE test_while1(IN insertCount INT)
BEGIN
	DECLARE i INT DEFAULT 0;
	a:WHILE i<=insertCount DO
		SET i=i+1;
		IF MOD(i,2)!=0 THEN ITERATE a;
		END IF;
		
		INSERT INTO admin(username,`password`) VALUES(CONCAT('xiaohua',i),'0000');
		
	END WHILE a;
END $


CALL test_while1(100)$

/*

int i=0;
while(i<=insertCount){
	i++;
	if(i%2==0){
		continue;
	}
	插入
	
}

*/

测试

1、已知表stringcontent
其中字段:
id 自增长
content varchar(20)
向该表插入指定个数的,随机的字符串


CREATE TABLE IF NOT EXISTS stringcontent(
	id INT PRIMARY KEY AUTO_INCREMENT,
	content VARCHAR(20)
);

DROP PROCEDURE test_randstr_insert;

DELIMITER $
CREATE PROCEDURE test_randstr_insert(IN insertCount INT)
BEGIN
	DECLARE i INT DEFAULT 1;  #定义一个循环变量i,表示插入次数
	DECLARE str VARCHAR(26) DEFAULT 'abcdefghjklmnopqrstuvwxyz';
	DECLARE startIndex INT DEFAULT 1; #代表起始索引
	DECLARE len INT DEFAULT 1; #截取长度
	WHILE i<= insertCount DO
		SET startIndex = FLOOR(RAND()*26+1);
		SET len = FLOOR(RAND()*(26-startIndex+1)+1);
		INSERT INTO stringcontent(content) VALUES(SUBSTR(str,startIndex,len));
		SET i = i + 1;
	END WHILE;
END $

CALL test_randstr_insert(10);

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值