一、视图
1.1、概述
1.2、语法
1.2.1、创建
CREATE [OR REPLACE] VIEW 视图名称[(列名列表)] AS SELECT语句 [ WITH [CASCADED | LOCAL ] CHECK OPTION ]
1.2.2、查询
# 查看创建视图的语句:
SHOW CREATE VIEW 视图名称;
# 查看视图数据:
SELECT * FROM 视图名称 ...... ;
1.2.3、修改
方式一:CREATE [OR REPLACE] VIEW 视图名称[(列名列表)] AS SELECT语句 [WITH [CASCADED | LOCAL ] CHECK OPTION ]
方式二:ALTER VIEW 视图名称[(列名列表)] AS SELECT语句 [WITH [ CASCADED | LOCAL ] CHECK OPTION ]
1.2.4、删除
DROP VIEW [IF EXISTS] 视图名称 [,视图名称] ...
1.3、案例
1.3.1、创建视图
create or replace view stu_v_1 as select id,name from student where id <= 10;
1.3.2、查询视图
show create view stu_v_1\G;
select * from stu_v_1;
select * from stu_v_1 where id < 3;
1.3.3、修改视图
create or replace view stu_v_1 as select id,name,no from student where id <= 10;
alter view stu_v_1 as select id,name from student where id <= 10;
1.3.4、删除视图
drop view if exists stu_v_1;
1.3.5、测试往视图中插入数据
create or replace view stu_v_1 as select id,name,no from student where id <= 10;
insert into stu_v_1 values(5,'张无忌','2000100105');
insert into stu_v_1 values(11,'小昭','2000100111');
结果分析:因为我们在创建视图的时候,指定的条件为 id<=10, id为11的数据,是不符合条件的,所以没有查询出来,但是这条数据确实是已经成功的插入到了基表中。如果我们定义视图时,如果指定了条件,然后我们在插入、修改、删除数据时,是否可以做到必须满足条件才能操作,否则不能够操作呢? 答案是可以的,这就需要借助于视图的检查选项了。
1.4、检查选项
1.4.1、CASCADED
级联。
1.4.2、LOCAL
本地
1.5、视图的更新
-
聚合函数或窗口函数( SUM() 、 MIN() 、 MAX() 、 COUNT() 等)
-
DISTINCT
-
GROUP BY
-
HAVING
-
UNION 或者 UNION ALL
1.6、示例演示
create view stu_v_count as select count(*) from student;
1.7、视图的作用
1.7.1、简单
1.7.2、安全
1.7.3、数据独立
1.8、案例
1.8.1、案例一
需求:为了保证数据库表的安全性,开发人员在操作tb_user表时,只能看到的用户的基本字段,屏蔽手机号和邮箱两个字段。
create view tb_user_view as select id,name,profession,age,gender,status,createtime from tb_user;
select * from tb_user_view;
1.8.2、案例二
需求:查询每个学生所选修的课程(三张表联查),这个功能在很多的业务中都有使用到,为了简化操作,定义一个视图。
create view tb_stu_course_view as select s.name student_name , s.no student_no ,c.name course_name from student s, student_course sc , course c where s.id = sc.studentid and sc.courseid = c.id;
select * from tb_stu_course_view;
二、存储过程
2.1、概述
2.2、特点
- 封装,复用:可以把某一业务SQL封装在存储过程中,需要用到的时候直接调用即可;
- 可以接收参数,也可以返回数据;
- 减少网络交互效率提升:如果涉及到多条SQL,每执行一次都是一次网络传输。 而如果封装在存储过程中,我们只需要网络交互一次可能就可以了
2.3、基本用法
2.3.1、创建语法
CREATE PROCEDURE 存储过程名称 ([ 参数列表 ])
BEGIN
-- SQL语句
END ;
2.3.2、调用语法
CALL 名称 ([ 参数 ]);
2.3.3、查看语法
SELECT * FROM INFORMATION_SCHEMA.ROUTINES WHERE ROUTINE_SCHEMA = 'xxx'; -- 查询指定数据库的存储过程及状态信息
SHOW CREATE PROCEDURE 存储过程名称 ; -- 查询某个存储过程的定义
2.3.4、删除语法
DROP PROCEDURE [ IF EXISTS ] 存储过程名称;
2.3.5、注意事项
2.4、案例
2.4.1、创建存储过程
# 创建简单存储过程
create procedure p1()
begin
select count(*) from student;
end;
# 创建存储过程(标准语法)
DELIMITER $$
USE `20230520_vhr`$$
DROP PROCEDURE IF EXISTS `listAllDepartment`$$
CREATE DEFINER=`root`@`localhost` PROCEDURE `listAllDepartment`()
BEGIN
SELECT * FROM department;
END$$
DELIMITER ;
# 创建复杂存储过程
DELIMITER $$
USE `20230520_vhr`$$
DROP PROCEDURE IF EXISTS `createDepartment`$$
CREATE DEFINER=`root`@`localhost` PROCEDURE `createDepartment`(IN departmentName VARCHAR(32),IN parentId INT,IN enabled TINYINT,OUT result INT,OUT result2 INT)
BEGIN
DECLARE dId INT;
DECLARE parentDepartmentPath VARCHAR(64);
INSERT INTO department SET NAME = departmentName,parent_id = parentId,enabled = enabled;
SELECT ROW_COUNT() INTO result;
SELECT LAST_INSERT_ID() INTO did;
SET result2 = dId;
SELECT department_path INTO parentDepartmentPath FROM department WHERE id = parent_id;
UPDATE department SET department_path = CONCAT(parentDepartmentPath,'.',dId) WHERE id = dId;
UPDATE department SET is_parent = TRUE WHERE id = parent_id;
END$$
DELIMITER ;
2.4.2、调用存储过程
call p1();
2.4.3、查看存储过程
select * from information_schema.ROUTINES where ROUTINE_SCHEMA = 'itheima';
show create procedure p1;
2.4.4、删除存储过程
drop procedure if exists p1;
2.5、变量
2.5.1、系统变量
SHOW [ SESSION | GLOBAL ] VARIABLES ; -- 查看所有系统变量
SHOW [ SESSION | GLOBAL ] VARIABLES LIKE 'xxx'; -- 可以通过LIKE模糊匹配方式查找变量
SELECT @@[SESSION | GLOBAL] 系统变量名; -- 查看指定变量的值
设置系统变量
SET [SESSION | GLOBAL] 系统变量名 = 值 ;
SET @@[SESSION | GLOBAL] 系统变量名 = 值 ;
注意:如果没有指定SESSION/GLOBAL,默认是SESSION,会话变量。
全局变量(GLOBAL): 全局变量针对于所有的会话。
会话变量(SESSION): 会话变量针对于单个会话,在另外一个会话窗口就不生效了。
案例演示
-- 查看系统变量
show session variables;
show session variables like 'auto%';
show global variables like 'auto%';
select @@global.autocommit;
select @@session.autocommit;
-- 设置系统变量
set session autocommit = 1;
insert into course(id, name) VALUES (6, 'ES');
set global autocommit = 0;
select @@global.autocommit;
2.5.2、用户自定义变量
方式一:
SET @var_name = expr [, @var_name = expr] ... ;
SET @var_name := expr [, @var_name := expr] ... ;
方式二:
SELECT @var_name := expr [, @var_name := expr] ... ;
SELECT 字段名 INTO @var_name FROM 表名;
(2)、使用
SELECT @var_name ;
注意: 用户定义的变量无需对其进行声明或初始化,只不过获取到的值为NULL。
(3)、案例
-- 赋值
set @myname = 'itcast';
set @myage := 10;
set @mygender := '男',@myhobby := 'java';
select @mycolor := 'red';
select count(*) into @mycount from tb_user;
-- 使用
select @myname,@myage,@mygender,@myhobby;
select @mycolor , @mycount;
2.5.3、局部变量
DECLARE 变量名 变量类型 [DEFAULT ... ] ;
变量类型就是数据库字段类型:INT、BIGINT、CHAR、VARCHAR、DATE、TIME等。
SET 变量名 = 值 ;
SET 变量名 := 值 ;
SELECT 字段名 INTO 变量名 FROM 表名 ... ;
(3)、案例
-- 声明局部变量 - declare
-- 赋值
create procedure p2()
begin
declare stu_count int default 0;
select count(*) into stu_count from student;
select stu_count;
end;
-- 调用
call p2();
2.6、if
2.6.1、概述
IF 条件1 THEN
.....
ELSEIF 条件2 THEN -- 可选
.....
ELSE -- 可选
.....
END IF;
2.6.2、案例
需求:根据定义的分数score变量,判定当前分数对应的分数等级
-
score >= 85 分,等级为优秀
-
score >= 60 分 且 score < 85 分,等级为及格
-
score < 60 分,等级为不及格
create procedure p3()
begin
declare score int default 58;
declare result varchar(10);
if score >= 85 then
set result := '优秀';
elseif score >= 60 then
set result := '及格';
else
set result := '不及格';
end if;
select result;
end;
call p3();
分析:上述的需求我们虽然已经实现了,但是也存在一些问题,比如:score 分数我们是在存储过程中定义死的,而且最终计算出来的分数等级,我们也仅仅是最终查询展示出来而已。 那么我们能不能,把score分数动态的传递进来,计算出来的分数等级是否可以作为返回值返回呢? 答案是肯定的,我们可以通过接下来所讲解的参数来解决上述的问题。
2.7、参数
2.7.1、概述
2.7.2、用法
CREATE PROCEDURE 存储过程名称 ([ IN/OUT/INOUT 参数名 参数类型 ])
BEGIN
-- SQL语句
END ;
2.7.3、案例一
需求:根据传入参数score,判定当前分数对应的分数等级,并返回
-
score >= 85 分,等级为优秀
-
score >= 60 分 且 score < 85 分,等级为及格
-
score < 60 分,等级为不及格
-- 创建存储过程,输入参数为score,输出参数为result
create procedure p4(in score int, out result varchar(10))
begin
if score >= 85 then
set result := '优秀';
elseif score >= 60 then
set result := '及格';
else
set result := '不及格';
end if;
end;
-- 定义用户变量 @result来接收返回的数据, 用户变量可以不用声明
call p4(18, @result);
select @result;
2.7.4、案例二
需求:将传入的200分制的分数,进行换算,换算成百分制,然后返回。
-- 创建存储过程
create procedure p5(inout score double)
begin
set score := score * 0.5;
end;
-- 调用存储过程
set @score = 198;
call p5(@score);
select @score;
2.8、case
2.8.1、概述
# 语法一
-- 含义:当case_value的值为 when_value1时,执行statement_list1,当值为 when_value2时,执行statement_list2,否则就执行statement_list
CASE case_value
WHEN when_value1 THEN statement_list1
[ WHEN when_value2 THEN statement_list2] ...
[ ELSE statement_list ]
END CASE;
# 语法二
-- 含义:当条件search_condition1成立时,执行statement_list1,当条件search_condition2成立时,执行statement_list2,否则就执行statement_list
CASE
WHEN search_condition1 THEN statement_list1
[WHEN search_condition2 THEN statement_list2] ...
[ELSE statement_list]
END CASE;
2.8.2、案例
需求:根据传入的月份,判定月份所属的季节(要求采用case结构)
-
1-3 月份,为第一季度
-
4-6 月份,为第二季度
-
7-9 月份,为第三季度
-
10-12 月份,为第四季度
-- 创建存储过程
create procedure p6(in month int,out result varchar(10))
begin
case
when month >= 1 and month <= 3 then
set result := '第一季度';
when month >= 4 and month <= 6 then
set result := '第二季度';
when month >= 7 and month <= 9 then
set result := '第三季度';
when month >= 10 and month <= 12 then
set result := '第四季度';
else
set result := '非法参数';
end case ;
select concat('您输入的月份为: ',month, ', 所属的季度为: ',result);
end;
-- 调用存储过程
call p6(10,@result);
---------------------------------------------------------------------
-- 创建存储过程
create procedure p7(in month int)
begin
declare result varchar(10);
case
when month >= 1 and month <= 3 then
set result := '第一季度';
when month >= 4 and month <= 6 then
set result := '第二季度';
when month >= 7 and month <= 9 then
set result := '第三季度';
when month >= 10 and month <= 12 then
set result := '第四季度';
else
set result := '非法参数';
end case ;
select concat('您输入的月份为: ',month, ', 所属的季度为: ',result);
end;
-- 调用存储过程
call p7(16);
注意事项:如果判定条件有多个,多个条件之间,可以使用 and 或 or 进行连接
2.9、while
2.9.1、概述
-- 先判定条件,如果条件为true,则执行逻辑,否则,不执行逻辑
WHILE 条件 DO
SQL逻辑...
END WHILE;
2.9.2、案例
需求:计算从1累加到n的值,其中n为传入的参数值(while循环实现)
-- A. 定义局部变量, 记录累加之后的值;
-- B. 每循环一次, 就会对n进行减1 , 如果n减到0, 则退出循环
create procedure p8(in n int)
begin
declare total int default 0;
while n > 0 do
set total := total + n;
set n := n - 1;
end while;
select total;
end;
-- 调用存储过程
call p8(100);
2.10、repeat
2.10.1、概述
-- 先执行一次逻辑,然后判定UNTIL条件是否满足,如果满足,则退出。如果不满足,则继续下一次循环
REPEAT
SQL逻辑...
UNTIL 条件
END REPEAT;
2.10.2、案例
需求:计算从1累加到n的值,其中n为传入的参数值(repeat实现)
-- A. 定义局部变量, 记录累加之后的值;
-- B. 每循环一次, 就会对n进行-1 , 如果n减到0, 则退出循环
-- 创建存储过程
create procedure p9(in n int)
begin
declare total int default 0;
repeat
set total := total + n;
set n := n - 1;
until n <= 0
end repeat;
select total;
end;
-- 调用存储过程
call p9(10);
call p9(100);
2.11、loop
2.11.1、概述
[begin_label:] LOOP
SQL逻辑...
END LOOP [end_label];
2.11.2、案例一
需求:计算从1累加到n的值,n为传入的参数值
-- A. 定义局部变量, 记录累加之后的值;
-- B. 每循环一次, 就会对n进行-1 , 如果n减到0, 则退出循环 ----> leave xx
create procedure p10(in n int)
begin
declare total int default 0;
sum:loop
if n <= 0 then
leave sum;
end if;
set total := total + n;
set n := n - 1;
end loop sum;
select total;
end;
call p10(100);
2.11.3、案例二
需求:计算从1到n之间的偶数累加的值,n为传入的参数值
-- A. 定义局部变量, 记录累加之后的值;
-- B. 每循环一次, 就会对n进行-1 , 如果n减到0, 则退出循环 ----> leave xx
-- C. 如果当次累加的数据是奇数, 则直接进入下一次循环. --------> iterate xx
create procedure p11(in n int)
begin
declare total int default 0;
sum:loop
if n <= 0 then
leave sum;
end if;
if n % 2 = 1 then
set n := n - 1;
iterate sum;
end if;
set total := total + n;
set n := n - 1;
end loop sum;
select total;
end;
call p11(100);
2.12、游标
2.12.1、概述
# 声明游标
DECLARE 游标名称 CURSOR FOR 查询语句 ;
# 打开游标
OPEN 游标名称 ;
# 获取游标记录
FETCH 游标名称 INTO 变量 [, 变量 ] ;
# 关闭游标
CLOSE 游标名称 ;
2.12.2、案例
需求:根据传入的参数uage,来查询用户表tb_user中所有的用户年龄小于等于uage的用户姓名(name)和专业(profession),并将用户的姓名和专业插入到所创建的一张新表 (id,name,profession)中
-- 逻辑:
-- A. 声明游标, 存储查询结果集
-- B. 准备: 创建表结构
-- C. 开启游标
-- D. 获取游标中的记录
-- E. 插入数据到新表中
-- F. 关闭游标
create procedure p12(in uage int)
begin
declare uname varchar(100);
declare upro varchar(100);
declare u_cursor cursor for select name,profession from tb_user where age <= uage;
drop table if exists tb_user_pro;
create table if not exists tb_user_pro(
id int primary key auto_increment,
name varchar(100),
profession varchar(100)
);
open u_cursor;
while true do
fetch u_cursor into uname,upro;
insert into tb_user_pro values (null, uname, upro);
end while;
close u_cursor;
end;
call p12(30);
但是此时,tb_user_pro表结构及其数据都已经插入成功了,我们可以直接刷新表结构,检查表结构中的数据。
上述的功能虽然我们实现了,但是逻辑并不完善,而且程序执行完毕,获取不到数据,数据库还报错。 接下来,我们就需要来完成这个存储过程,并且解决这个问题。要想解决这个问题,就需要通过MySQL中提供的条件处理程序 Handler 来解决。
2.13、条件处理程序
2.13.1、概述
DECLARE handler_action HANDLER FOR condition_value [, condition_value] ... statement ;
handler_action 的取值:
CONTINUE: 继续执行当前程序
EXIT: 终止执行当前程序
condition_value 的取值:
SQLSTATE sqlstate_value: 状态码,如 02000
SQLWARNING: 所有以01开头的SQLSTATE代码的简写
NOT FOUND: 所有以02开头的SQLSTATE代码的简写
SQLEXCEPTION: 所有没有被SQLWARNING 或 NOT FOUND捕获的SQLSTATE代码的简写
2.13.2、案例
需求:同2.12.2中的需求,本案例是对其的完善
-- 逻辑:
-- A. 声明游标, 存储查询结果集
-- B. 准备: 创建表结构
-- C. 开启游标
-- D. 获取游标中的记录
-- E. 插入数据到新表中
-- F. 关闭游标
create procedure p12(in uage int)
begin
declare uname varchar(100);
declare upro varchar(100);
declare u_cursor cursor for select name,profession from tb_user where age <= uage;
-- 声明条件处理程序 : 当SQL语句执行抛出的状态码为02000时,将关闭游标u_cursor,并退出
declare exit handler for SQLSTATE '02000' close u_cursor;
-- 建表
drop table if exists tb_user_pro;
create table tb_user_pro(
id int primary key auto_increment,
name varchar(100),
profession varchar(100)
);
open u_cursor;
while true do
fetch u_cursor into uname,upro;
insert into tb_user_pro values (null, uname, upro);
end while;
close u_cursor;
end;
call p11(30);
create procedure p13(in uage int)
begin
declare uname varchar(100);
declare upro varchar(100);
declare u_cursor cursor for select name,profession from tb_user where age <= uage;
-- 声明条件处理程序 : 当SQL语句执行抛出的状态码为02开头时,将关闭游标u_cursor,并退出
declare exit handler for not found close u_cursor;
drop table if exists tb_user_pro;
create table if not exists tb_user_pro(
id int primary key auto_increment,
name varchar(100),
profession varchar(100)
);
open u_cursor;
while true do
fetch u_cursor into uname,upro;
insert into tb_user_pro values (null, uname, upro);
end while;
close u_cursor;
end;
call p13(30);
三、存储函数
3.1、概述
CREATE FUNCTION 存储函数名称 ([ 参数列表 ])
RETURNS type [characteristic ...]
BEGIN
-- SQL语句
RETURN ...;
END ;
characteristic说明:
DETERMINISTIC:相同的输入参数总是产生相同的结果
NO SQL :不包含 SQL 语句
READS SQL DATA:包含读取数据的语句,但不包含写入数据的语句
3.2、案例
需求:计算从1累加到n的值,n为传入的参数值
create function fun1(n int)
returns int deterministic
begin
declare sum int default 0;
while n > 0 do
set sum := sum + n;
set n := n - 1;
end while;
return sum;
end;
select fun1(100);
注意事项:在mysql8.0版本中binlog默认是开启的,一旦开启了,mysql就要求在定义存储过程时,需要指定characteristic特性,否则就会报如下错误
四、触发器
4.1、概述
4.2、语法
4.2.1、创建
CREATE TRIGGER trigger_name
BEFORE/AFTER INSERT/UPDATE/DELETE
ON tbl_name FOR EACH ROW -- 行级触发器
BEGIN
trigger_stmt ;
END;
4.2.2、查看
SHOW TRIGGERS ;
4.2.3、删除
DROP TRIGGER [schema_name.]trigger_name ; -- 如果没有指定 schema_name,默认为当前数据库 。
4.3、案例
需求:通过触发器记录 tb_user 表的数据变更日志,将变更日志插入到日志表tb_user_logs中, 包含增加, 修改 , 删除 ;
4.3.1、tb_user_logs表准备
-- 准备工作 : 日志表 tb_user_logs
create table tb_user_logs (
id int(11) not null auto_increment,
operation varchar(20) not null comment '操作类型, insert/update/delete',
operate_time datetime not null comment '操作时间',
operate_id int(11) not null comment '操作的ID',
operate_params varchar(500) comment '操作参数',
primary key(`id`)
)engine=innodb default charset=utf8;
4.3.2、创建数据插入触发器
-- 创建触发器
create trigger tb_user_insert_trigger
after insert on tb_user for each row
begin
insert into tb_user_logs(id, operation, operate_time, operate_id, operate_params)
VALUES
(
null,
'insert',
now(),
new.id,
concat('插入的数据内容为:id=',new.id,',name=',new.name, ', phone=', NEW.phone, ', email=', NEW.email, ',profession=', NEW.profession));
end;
-- 测试
-- 查看触发器
show triggers;
-- 插入数据到tb_user表
insert into tb_user(id, name, phone, email, profession, age, gender, status,createtime)
VALUES
(26,'三皇子','18809091212','erhuangzi@163.com','软件工程',23,'1','1',now());
-- 观察tb_user表和tb_user_logs表中的数据
4.3.3、创建更新触发器
-- 创建触发器
create trigger tb_user_update_trigger
after update on tb_user for each row
begin
insert into tb_user_logs(id, operation, operate_time, operate_id, operate_params)
VALUES
(
null,
'update',
now(),
new.id,
concat('更新之前的数据: id=',old.id,',name=',old.name, ', phone=',old.phone, ',email=', old.email, ', profession=', old.profession,' | 更新之后的数据: id=', new.id,',name=',new.name, ', phone=',NEW.phone, ', email=', NEW.email, ', profession=', NEW.profession)
);
end;
-- 测试
-- 查看触发器
show triggers;
-- 更新数据
update tb_user set profession = '会计' where id = 1;
update tb_user set profession = '会计' where id <= 5;
-- 观察tb_user表和tb_user_logs表中的数据变化
4.3.4、创建删除触发器
-- 创建删除触发器
create trigger tb_user_delete_trigger
after delete on tb_user for each row
begin
insert into tb_user_logs(id, operation, operate_time, operate_id, operate_params)
VALUES
(null, 'delete', now(), old.id,
concat('删除之前的数据: id=',old.id,',name=',old.name, ', phone=',
old.phone, ', email=', old.email, ', profession=', old.profession));
end;
-- 测试
-- 查看触发器
show triggers;
-- 观察tb_user表和tb_user_logs表中的数据