视图、存储过程和触发器

一、视图

1.1 简介

        视图(View)是一种虚拟存在的表。视图中的数据并不在数据库中实际存在,行和列数据来自定义视图的查询中使用的表,并且是在使用视图时动态生成的。

        通俗的讲,视图只保存了查询的 sql 逻辑,不保存查询结果。所以我们在创建视图的时候,主要的工作就落在创建这条 sql 查询语句上。

1.2 语法

1.2.1 创建

CREATE [OR REPLACE] VIEW 视图名称[(列名列表)] AS SELECT语句 [ WITH [CASCADED | LOCAL ] CHECK OPTION ]

        根据 student 表创建一个视图,sql 语句如下所示:

create or replace view stu_v_1 as select id,name from student where id<10;

1.2.2 查询

-- 查看创建视图语句
SHOW CREATE VIEW 视图名称;

-- 查看视图数据
SELECT * FROM 视图名称 ...... ;

        查看我们刚才创建的 stu_v_1 视图的数据,sql 语句如下所示:

select * from stu_v_1;

1.2.3 修改

-- 方式一,和创建的方式一致
CREATE [OR REPLACE] VIEW 视图名称[(列名列表)] AS SELECT语句 [ WITH [ CASCADED | LOCAL ] CHECK OPTION ]

-- 方式二
ALTER VIEW 视图名称[(列名列表)] AS SELECT语句 [ WITH [ CASCADED |LOCAL ] CHECK OPTION ]

         修改我们刚才创建的 stu_v_1 视图的数据,sql 语句如下所示:

-- 方式一
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.2.4 删除

DROP VIEW [IF EXISTS] 视图名称 [,视图名称] ...

        删除我们刚才创建的 stu_v_1 视图的数据,sql 语句如下所示:

drop view if exists stu_v_1;

1.3 插入数据

        上面我们演示了视图应该如何创建、查询、修改、删除,那么我们能不能通过视图来插入或者更新数据呢? 接下来,做一个测试。

-- 创建视图
create or replace view stu_v_1 as select id,name from student where id <= 10 ;

-- 查询视图数据
select * from stu_v_1;

-- 插入数据
insert into stu_v_1 values(6,'Tom');

-- 插入数据
insert into stu_v_1 values(17,'Tom22');

        执行上述的 sql,我们会发现,id617 的数据都是可以成功插入的。 但是我们执行查询时查询出来的数据,却没有 id17 的记录。

        因为我们在创建视图的时候,指定的条件为 id<=10id 17 的数据是不符合条件的,所以没有查询出来,但是这条数据确实是已经成功的插入到了基表中。如下

        如果我们定义视图时指定了条件,然后我们在插入、修改、删除数据时,是否可以做到必须满足条件才能操作,否则不能够操作呢? 答案是可以的,这就需要借助于视图的检查选项了。

1.4 检查选项

        当使用 with check option 子句创建视图时,MySQL 会通过视图检查正在更改的每个行,例如插入、更新和删除,以使其符合视图的定义。

        MySQL 允许基于另一个视图创建视图,它还会检查依赖视图中的规则以保持一致性。为了确定检查的范围,mysql 提供了两个选项: cascaded local,默认值为 cascaded

         首先执行下面的 sql 语句,创建视图并插入数据

-- 创建视图
create or replace view stu_v_1 as select id,name from student where id <= 20 with cascaded check option;

-- 插入数据
insert into stu_v_1 values(6,'Tom');

-- 插入数据
insert into stu_v_1 values(30,'Tom22');

        可以看到插入 id 30 的这条记录报错了,因为它违背了我们创建视图的条件,我们规定了要检查 id < 20 的数据。 

1.4.1 cascaded 级联

二级关联:

        假设 v2 视图是基于 v1 视图的,如果在 v2 视图创建的时候指定了检查选项为 cascaded,但是 v1 视图创建时未指定检查选项。 则在执行检查时,不仅会检查 v2,还会级联检查 v2 的关联视图 v1

        首先执行下面的 sql 语句

-- 创建视图
create or replace view stu_v_1 as select id,name from student where id <= 20 

-- 插入数据
insert into stu_v_1 values(5,'Tom');

-- 插入数据
insert into stu_v_1 values(25,'Tom22');

        数据是可以插入成功的,因为我们在创建视图的时候并没有加入检查选项,那么 mysql 就不会去检查我们的增删改操作。 

        再创建一个视图 stu_v_2 ,它是基于 stu_v_1 视图创建的,sql 语句如下:

-- 创建视图
create or replace view stu_v_2 as select id,name from stu_v_1 where id >=10 with cascaded check option; 

         插入一条数据,如下

insert into stu_v_2 values(7,'Tom');

        并没有插入成功,因为 7 不大于 10 ,不满足条件,所以此时就会报错,违背了检查选项。 

        再插入一条数据,如下:

insert into stu_v_2 values(26,'Tom');

        并没有插入成功,因为它不仅要去检查当前视图,还需要去检查他所依赖的视图,他去检查 stu_v_1 视图时,发现 26 是大于 20 的,无法满足条件,无法插入成功。

        最后我们再插入一条数据,如下: 

insert into stu_v_2 values(15,'Tom');

         我们发现执行成功了,首先判断 15 是否大于 10,其次再判断 15 是否小于等于 20,都满足条件就可以插入成功了。

三级关联:

        此时我们再创建一个视图,这个视图是没有添加检查约束的,但是它所依赖的视图是有检查约束的,如下图,我们看看会发生什么情况

        再创建一个视图 stu_v_3,它是基于 stu_v_2 视图创建的且没有约束条件,sql 语句如下:

-- 创建视图
create or replace view stu_v_3 as select id,name from stu_v_2 where id <=15;

         执行下面的 sql 语句

-- 插入数据
insert into stu_v_3 values(11,'Tom');

-- 插入数据
insert into stu_v_3 values(17,'Tom');

-- 插入数据
insert into stu_v_3 values(28,'Tom');

        前两条插入成功了,第三条插入失败了。我们分析下原因

        当插入 id = 11 的数据时,由于 stu_v_3 没有约束条件,此时去找它所依赖的 stu_v_2 视图,id=11 满足 stu_v_2 的约束条件,最后再去找 stu_v_2 所依赖的 stu_v_1 视图,发现也满足它的约束条件,所以可以插入成功。

        当插入 id = 17 的数据时,由于 stu_v_3 没有约束条件,此时去找它所依赖的 stu_v_2 视图,id=17 满足 stu_v_2 的约束条件,最后再去找 stu_v_2 所依赖的 stu_v_1 视图,发现也满足它的约束条件,所以可以插入成功。

        当插入 id = 28 的数据时,由于 stu_v_3 没有约束条件,此时去找它所依赖的 stu_v_2 视图,id=28 满足 stu_v_2 的约束条件,最后再去找 stu_v_2 所依赖的 stu_v_1 视图,发现违背了  stu_v_1 id <= 20 的条件,所以就没有插入成功。

1.4.2 local 本地

二级关联:

        v2 视图是基于 v1 视图的,如果在 v2 视图创建的时候指定了检查选项为 local ,但是 v1 视图创建时未指定检查选项。 则在执行检查时,只会检查 v2,不会检查 v2 的关联视图 v1

         首先执行下面的 sql 语句

-- 创建视图
create or replace view stu_v_4 as select id,name from student where id <= 20 

-- 插入数据
insert into stu_v_4 values(5,'Tom');

-- 插入数据
insert into stu_v_4 values(25,'Tom22');

        我们发现,数据是可以插入成功的,因为我们在创建视图的时候并没有加入检查选项,那么 mysql 就不会去检查我们的增删改操作。 

        再创建一个视图 stu_v_5 ,它是基于 stu_v_4 视图创建的,sql 语句如下:

-- 创建视图
create or replace view stu_v_5 as select id,name from stu_v_4 where id >=10 with local check option; 

         插入一条数据,如下

insert into stu_v_5 values(7,'Tom');

        并没有插入成功,因为 7 不大于 10 ,不满足条件,所以此时就会报错,违背了检查选项。 

        再插入一条数据,如下:

insert into stu_v_5 values(26,'Tom');

        插入成功,它先去检查当前视图的约束条件,发现没有问题,然后去检查 stu_v_1 视图的约束条件,发现没有约束条件。

三级关联:

         此时我们再创建一个视图,这个视图是没有添加检查约束的,但是它所依赖的视图是有本地约束的,如下图,我们看看会发生什么情况

        创建一个视图 stu_v_6,它是基于 stu_v_5 视图创建的且没有约束条件,sql 语句如下:

-- 创建视图
create or replace view stu_v_6 as select id,name from stu_v_5 where id <=15;

         执行下面的 sql 语句

-- 插入数据
insert into stu_v_6 values(11,'Tom');

-- 插入数据
insert into stu_v_6 values(17,'Tom');

-- 插入数据
insert into stu_v_6 values(8,'Tom');

        可以看到,前两条插入成功了,第三条插入失败了。我们分析下原因。

        当插入 id = 11 的数据时,由于 stu_v_6 没有约束条件,此时去找它所依赖的 stu_v_5 视图,id=11 满足 stu_v_5 的约束条件,最后再去找 stu_v_5 所依赖的 stu_v_4 视图,发现它没有约束,所以可以插入成功。

        当插入 id = 17 的数据时,由于 stu_v_6 没有约束条件,此时去找它所依赖的 stu_v_5 视图,id=17 满足 stu_v_5 的约束条件,最后再去找 stu_v_5 所依赖的 stu_v_4 视图,发现它没有约束,所以可以插入成功。

        当插入 id = 8 的数据时,由于 stu_v_6 没有约束条件,此时去找它所依赖的 stu_v_5 视图,id=8 不满足 stu_v_5 的约束条件,所以就没有插入成功。

1.5 视图更新

        如果想要使视图可以进行更新操作,那么视图中的行与基础表中的行之间必须存在一对一的关系。如果视图包含以下任何一项,则该视图不可更新:

        1、创建视图时使用了聚合函数或窗口函数(sum()、 min()、 max()、 count() 等)

        2、创建视图时使用了 distinct

        3、创建视图时使用了 group by

        4、创建视图时使用了 having 

        5、创建视图时使用了 union 或者 union all 

案例演示

        创建一个视图,命令如下:

create view stu_v_count as select count(*) from student;

        上述的视图中,就只有一个单行单列的数据,如果我们对这个视图进行更新或插入的,将会报错。

1.6 视图作用

        简单,视图不仅可以简化用户对数据的理解,也可以简化他们的操作。那些被经常使用的查询可以被定义为视图,从而使得用户不必为以后的操作每次指定全部的条件。

        安全,数据库可以授权,但不能授权到数据库特定行和特定的列上。通过视图用户只能查询和修改他们所能见到的数据。

        数据独立,视图可帮助用户屏蔽真实表结构变化带来的影响。

1.7 案例

        1、为了保证数据库表的安全性,开发人员在操作 tb_user 表时,只能看到的用户的基本字段,屏蔽手机号和邮箱两个字段。

create or replace  view tb_user_view as select id,name,profession,age,gender,status,createtime from tb_user;

        2、查询每个学生所选修的课程(三张表联查),这个功能在很多的业务中都有使用到,为了简化操作,定义一个视图。

create or replace view s_c_view as select s.name,s.no,c.name from student s,course c,student_course sc where s.id = sc.studentid and c.id = sc.courseid; 

二、存储过程

2.1 简介

        存储过程是事先经过编译并存储在数据库中的一段 sql 语句的集合,调用存储过程可以简化应用开发人员的很多工作,减少数据在数据库和应用服务器之间的传输,对于提高数据处理的效率是有好处的。 存储过程思想上很简单,就是数据库 sql 语言层面的代码封装与重用。

2.2 特点

        1、封装、复用。可以把某一业务 sql 封装在存储过程中,需要用到 的时候直接调用即可。

        2、可以接收参数,也可以返回数据。在存储过程中,可以传递参数,也可以接收返回值。

        3、较少网络交互,提升效率。如果涉及到多条 sql,每执行一次都是一次网络传输。 而如果封装在存储过程中,我们只需要网络交互一次可能就可以了。

2.3 基本语法

2.3.1 创建

create procedure 存储过程名称 ([参数列表])

begin

    -- sql 语句

end;

        创建一个查询记录数的存储过程,语句如下:

create procedure p1()

begin

    select count(*) from student;

end;

2.3.2 调用

call 存储过程名称([参数])

        调用刚才创建的 p1 存储过程,如下:

call p1();

2.3.3 查看

-- 查询指定数据库的存储过程及状态信息
select * from INFORMATION_SCHEMA.ROUTINES where ROUTINE_SCHEMA = 'xxx';

-- 查询某个存储过程的定义
SHOW CREATE PROCEDURE 存储过程名称 ;

        查看刚才创建的 p1 存储过程的相关信息

select * from information_schema.ROUTINES where ROUTINE_SCHEMA = 'itheima';

2.3.4 删除

DROP PROCEDURE [ IF EXISTS ] 存储过程名称;

        删除刚才创建的存储过程 p1 ,如下:

2.4 特别注意

        在命令行中执行创建存储过程的 sql 时,需要通过关键字 delimiter 指定 sql 语句的结束符。我们来模拟下这种场景。

        当我们将创建存储过程的语句粘贴到命令行进行执行的时候,发现执行报错了,因为 mysql 默认是以 ; 来当作命令的结束符的,当一个存储过程里面出现多个 ; 的时候,它就执行报错了。

        此时我们可以使用 delimiter 关键字来定义 sql 语句的结束符,

-- 定义 $$ 为 sql 语句的结束符
-- 执行这个命令的时候千万别写分号,要不结束符就变成 $$; 了
delimiter $$

-- 定义 aaa 为 sql 语句的结束符
delimiter aaa

2.5 变量

        在 MySQL 中变量分为三种类型:系统变量、用户定义变量、局部变量。

2.5.1 系统变量

        系统变量是 MySQL 服务器提供,不是用户定义的,属于服务器层面。分为 global 全局变量和 session 会话变量。

        查看系统变量,命令如下:

-- 查看所有系统变量
SHOW [ SESSION | GLOBAL ] VARIABLES ; 

-- 可以通过LIKE模糊匹配方式查找变量
SHOW [ SESSION | GLOBAL ] VARIABLES LIKE '......'; 

-- 查看指定变量的值
SELECT @@[SESSION | GLOBAL] 系统变量名; 

        设置系统变量,命令如下:

SET [ SESSION | GLOBAL ] 系统变量名 = 值 ;

SET @@[SESSION | GLOBAL] 系统变量名 = 值 ;

注意:

        1、如果没有指定 session / global,默认是会话变量 session

        2、mysql 服务重新启动之后,所设置的全局参数会失效,要想不失效,可以在 /etc/my.cnf 中配置。

        3、global 全局变量:全局变量针对于所有的会话。

        4、session 会话变量:会话变量针对于单个会话,在另外一个会话窗口就不生效了。

查看系统变量示例:

        1、查看所有的系统变量

show session variables ;

        2、查看 auto 开头的系统变量

show session variables like 'auto%';
show global variables like 'auto%';

         3、查看知道确切名称的系统变量

select @@global.autocommit;
select @@session.autocommit;

设置系统变量示例:

        1、将事务的提交方式改为手动,命令如下:

set session autocommit = 0;

        设置成功之后我们来验证下,可以看到设置成功了

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 表名;

        使用的命令很简单,如下所示:

SELECT @var_name ;

赋值示例:

        1、使用 set 命令设置用户自定义变量,命令如下:

set @myname = 'itcast';
set @myage :=10;
set @mygender := '男',@myhobby := 'java';

        2、使用 select 命令设置成系统变量,命令如下:

select @mycolor := 'red';

-- 将 select count(*) from tb_user 的结果赋值给 @mycount 变量
select count(*) into @mycount from tb_user;

查询示例:

        查询刚才创建的用户自定义变量的值,命令如下:

select @myname,@myage,@mygender,@myhobby,@mycolor , @mycount;

注意:

        用户定义的变量无需对其进行声明或初始化,只不过获取到的值为 null

2.5.3 局部变量

        局部变量是根据需要定义的,在局部生效的变量,访问之前,需要使用 declare 关键字进行声明。可用作存储过程内的局部变量和输入参数,局部变量的范围是在其内声明的 begin.......end  块。

声明语法:

        变量类型就是数据库字段类型:INTBIGINTCHARVARCHARDATETIME 等。

DECLARE 变量名 变量类型 [DEFAULT ... ] ;

赋值语法:

SET 变量名 = 值 ;
SET 变量名 := 值 ;
SELECT 字段名 INTO 变量名 FROM 表名 ... ;

示例演示:

        创建一个存储过程 p2,在创建的过程中创建了一个 stu_count 的局部变量。

-- 声明局部变量 - declare
-- 赋值
create procedure p2()
begin

    declare stu_count int default 0;
    select count(*) into stu_count from student;
    select stu_count;

end;

        查询下存储过程 p2 所返回的数据,如下:

 2.6 if 

2.6.1 简介

        if 是用于做条件判断,具体的语法结构如下:

if 条件1 then
    .....
elseif 条件2 then      -- 可选
    .....
else
    .....              -- 可选
end if;

        在 if 条件判断的结构中,elseif 结构可以有多个,也可以没有。 else 结构可以有,也可以没有。

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 && score<80 then
		set result = '及格';
	else
		set result = '不及格';
	end if;
	select result;
end;


call p3();

        上述的需求我们虽然已经实现了,但是也存在一些问题,比如:score 分数我们是在存储过程中定义死的,而且最终计算出来的分数等级,我们也仅仅是最终查询展示出来而已。

        那么我们能不能把 score 分数动态的传递进来,计算出来的分数等级是否可以作为返回值返回呢? 答案是肯定的,我们可以通过接下来所讲解的参数来解决上述的问题。

2.7 参数

2.7.1 简介

        参数的类型,主要分为以下三种:INOUTINOUT。 具体的含义如下:

类型含义备注
IN该类参数作为输入,也就是需要调用时传入值默认
OUT该类参数作为输出,也就是该参数可以作为返回值
INOUT既可以作为输入参数,也可以作为输出参数

        用法如下:

CREATE PROCEDURE 存储过程名称 ([ IN/OUT/INOUT 参数名 参数类型 ])

BEGIN

    -- SQL语句

END ;

2.7.2 案例

        1、根据传入的分数 score ,判定当前分数对应的分数等级,并返回

        score >= 85 分,等级为优秀。

        score >= 60 分且 score < 85 分,等级为及格。

        score < 60 分,等级为不及格。

create procedure p4(in score int,out result varchar(10))
begin

	if score >=85 then
		set result = '优秀';
	elseif score>=60 && score<80 then
		set result = '及格';
	else
		set result = '不及格';
	end if;
end;

-- 调用的存储过程时,定义一个用户变量 @result 来接收返回的数据,用户变量可以不用声明
call p4(90,@result);

-- 查看用户变量的值
select @result;

        2、将传入的 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 有两种语法格式,如下:

-- 含义: 当 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 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;

select p7(4);

2.8.3 注意

        如果判定条件有多个,多个条件之间,可以使用 and or 进行连接。

2.9 while

2.9.1 简介

        while 循环是有条件的循环控制语句。满足条件后,再执行循环体中的 sql 语句。具体语法为:

-- 先判定条件,如果条件为 true,则执行逻辑,否则,不执行逻辑
WHILE 条件 DO

    SQL逻辑...

END WHILE;

2.9.2 案例

        计算从 1 累加到 n 的值,n 为传入的参数值。

-- A. 定义局部变量, 记录累加之后的值;
-- B. 每循环一次, 就会对 n 进行减 1 , 如果 n 减到 0, 则退出循环
create procedure p9(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 p9(10);

2.10 repeat

2.10.1 简介

        repeat 是有条件的循环控制语句, 当满足 until 声明的条件的时候,则退出循环 。具体语法为:

-- 先执行一次逻辑,然后判定 UNTIL 条件是否满足,如果满足,则退出。如果不满足,则继续下一次循环
REPEAT

    SQL逻辑...
    UNTIL 条件

END REPEAT;

2.10.2 案例

        计算从 1 累加到 n 的值,n 为传入的参数值。(使用 repeat 实现)

-- A. 定义局部变量, 记录累加之后的值;
-- B. 每循环一次, 就会对 n 进行-1 , 如果 n 减到 0, 则退出循环
create procedure p10(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 p10(10);

2.11 loop

2.11.1 简介

        loop 可以实现简单的循环,如果不在 sql 逻辑中增加退出循环的条件,可以用其来实现简单的死循环。 loop 可以配合一下两个语句使用:

        leave:配合循环使用,退出循环

        iterate:必须用在循环中,作用是跳过当前循环剩下的语句,直接进入下一次循环

[begin_label:] LOOP

    SQL逻辑...

END LOOP [end_label];
LEAVE label; -- 退出指定标记的循环体
ITERATE label; -- 直接进入下一次循环

        上述语法中出现的 begin_labelend_labellabel 指的都是我们所自定义的标记。 

2.11.2 案例

        1、计算从 1 累加到 n 的值,n 为传入的参数值(使用 loop 实现)。

create procedure p11(in n int)
begin
	declare total int default 0;
	sum:loop
        -- 首先判断 n 的值是否小于等于 0,如果满足则结束循环 
		if n<= 0 then 
			leave sum;
		end if;
        -- 执行累加操作
		set total := total + n;
		set n := n-1;
	end loop sum;
	select total;
end;

call p11(10);

        2、计算从 1 n 之间的偶数累加的值,n 为传入的参数值(使用 loop 实现)。

create procedure p12(in n int)
begin
	declare total int default 0;
	sum:loop
		if n<= 0 then 
			leave sum;
		end if;
		
        -- 如果当次累加的数据是奇数, 则直接进入下一次循环
		if n%2 !=0 then 
			set n := n-1;	
			iterate sum;
		end if;
		
		set total := total + n;
		set n := n-1;
	end loop sum;
	select total;
end;

call p12(10);

2.11 游标

2.11.1 简介

        游标(cursor)是用来存储查询结果集的数据类型 , 在存储过程和函数中可以使用游标对结果集进行循环的处理。游标的使用包括游标的声明、openfetch close,其语法分别如下。

        声明游标

DECLARE 游标名称 CURSOR FOR 查询语句 ;

        打开游标

OPEN 游标名称 ;

        获取游标记录

FETCH 游标名称 INTO 变量 [, 变量 ] ;

        关闭游标

CLOSE 游标名称 ;

2.11.2 案例

        根据传入的参数 uage,来查询用户表 tb_user 中所有的用户年龄小于等于 uage 的用户姓名 name 和专业 profession ,并将用户的姓名和专业插入到所创建的一张新表 tb_user_pro 中,其中 tb_user_pro 包含字段 idname profession

create procedure p13(in uage int)
begin
	declare uname varchar(10);
	declare upro varchar(10);
    -- A.、声明游标, 存储查询结果集 
	declare u_cursor cursor for select name,profession from tb_user where age < uage;
	-- B、准备: 创建表结构
	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)
	);
    -- C、开启游标
	open u_cursor;
    -- 循环操作
	while true  do
        -- D、获取游标中的记录
		fetch u_cursor into uname,upro;
        -- E. 插入数据到新表中
		insert into tb_user_pro values(null,uname,upro);
	end while;
    -- F. 关闭游标
	close u_cursor;
end;

        上述的存储过程在创建的时候不会报错,但是在调用的过程中会报错,之所以报错是因为上面的 while 循环中,并没有退出条件。当游标的数据集获取完毕之后,再次获取数据,就会报错,从而终止了程序的执行。 如下图:

        但是此时,tb_user_pro 表结构及其数据都已经插入成功了,我们可以直接刷新表结构,检查表结构中的数据。

        上述的功能,虽然我们实现了,但是逻辑并不完善,而且程序执行完毕,获取不到数据,数据库还报错。 接下来,我们就需要来完成这个存储过程,并且解决这个问题。

        要想解决这个问题,就需要通过 MySQL 中提供的条件处理程序 Handler 来解决。

2.12 条件处理程序

2.12.1 简介

        条件处理程序 handler 可以用来定义在流程控制结构执行过程中遇到问题时相应的处理步骤。具体语法为:

declare handler_action HANDLER FOR condition_value [, condition_value] ... statement ;


-- 其中 handler_action 的取值如下

    continue: 继续执行当前程序

    exit: 终止执行当前程序

-- 其中 condition_value 的取值如下

    SQLSTATE : 状态码,如 02000

    SQLWARNING: 所有以 01 开头的 SQLSTATE 代码的简写

    NOT FOUND: 所有以 02 开头的 SQLSTATE 代码的简写

    SQLEXCEPTION: 所有没有被 SQLWARNING 或 NOT FOUND 捕获的 SQLSTATE 代码的简写

2.12.2 案例

        我们继续来完成在上一小节提出的这个需求,并解决其中的问题。

        根据传入的参数 uage,来查询用户表 tb_user 中所有的用户年龄小于等于 uage 的用户姓名 name 和专业 profession ,并将用户的姓名和专业插入到所创建的一张新表 tb_user_pro 中,其中 tb_user_pro 包含字段 idname profession

        方式一:通过 SQLSTATE 指定具体的状态码,如下:

create procedure p13(in uage int)
begin
	declare uname varchar(10);
	declare upro varchar(10);
    -- A.、声明游标, 存储查询结果集 
	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
	-- B、准备: 创建表结构
	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)
	);
    -- C、开启游标
	open u_cursor;
    -- 循环操作
	while true  do
        -- D、获取游标中的记录
		fetch u_cursor into uname,upro;
        -- E. 插入数据到新表中
		insert into tb_user_pro values(null,uname,upro);
	end while;
    -- F. 关闭游标
	close u_cursor;
end;

        方式二:通过 SQLSTATE 的代码简写方式 NOT FOUND

create procedure p13(in uage int)
begin
	declare uname varchar(10);
	declare upro varchar(10);
    -- A.、声明游标, 存储查询结果集 
	declare u_cursor cursor for select name,profession from tb_user where age < uage;
	-- 声明条件处理程序 : 当 sql 语句执行抛出的状态码为 02000 时,将关闭游标 u_cursor,并退出
	declare exit handler for not found close u_cursor;
	-- B、准备: 创建表结构
	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)
	);
    -- C、开启游标
	open u_cursor;
    -- 循环操作
	while true  do
        -- D、获取游标中的记录
		fetch u_cursor into uname,upro;
        -- E. 插入数据到新表中
		insert into tb_user_pro values(null,uname,upro);
	end while;
    -- F. 关闭游标
	close u_cursor;
end;

        具体的错误状态码,可以参考官方文档:

         MySQL :: MySQL 8.0 Reference Manual :: 13.6.7.2 DECLARE ... HANDLER Statement

         MySQL :: MySQL 8.0 Error Reference :: 2 Server Error Message Reference

三、存储函数

3.1 简介

        存储函数是有返回值的存储过程,存储函数的参数只能是 in 类型的。具体语法如下:

CREATE FUNCTION 存储函数名称 ([ 参数列表 ])
RETURNS type [characteristic ...]
BEGIN
    -- SQL语句
    RETURN ...;
END ;

characteristic 说明:

        DETERMINISTIC:相同的输入参数总是产生相同的结果,即输入和输出参数相同

        NO SQL :不包含 sql 语句,即里面不包含 sql 语句。

        READS SQL DATA:包含读取数据的语句,但不包含写入数据的语句。

3.2 案例

        计算从 1 累加到 n 的值,n 为传入的参数值。

create function fun1(n int)
returns int deterministic
begin
	declare total int default 0;
	while n >0 do
		set total := total + n;
		set n := n-1;
	end while;
	return total;
end;

-- 调用的代码
select fun1(10);

        在 mysql8.0 版本中 binlog 默认是开启的,一旦开启了,mysql 就要求在定义存储过程时,需要指定 characteristic 特性,否则就会报如下错误:

3.3 总结

        存储函数用的相对来说比较少,因为存储函数能做的事情存储过程也可以做,而且存储函数存在一个弊端,必须得有返回值。

四、触发器

4.1 简介

        触发器是与表有关的数据库对象,指在 insert/update/delete 之前或之后,触发并执行触发器中定义的 sql 语句集合。触发器的这种特性可以协助应用在数据库端确保数据的完整性 , 日志记录,数据校验等操作 。

        使用别名 OLD NEW 来引用触发器中发生变化的记录内容,这与其他的数据库是相似的。现在触发器还只支持行级触发,不支持语句级触发。

触发器类型NEW 和 OLD
INSERT 型触发器NEW 表示将要或者已经新增的数据
UPDATE 型触发器OLD 表示修改之前的数据 , NEW 表示将要或已经修改后的数据
DELETE 型触发器OLD 表示将要或者已经删除的数据

        行级触发:假设此时执行了 update 语句,它影响了 5 行数据,那么此时行级触发器就会被触发 5 次,这就被称为行级触发器。

        语句级触发:假设此时执行了 update 语句,不管这条语句影响了多少行,我只触发一次,这就被称之为语句级触发器。

4.2 语法

        创建触发器的语法如下:

CREATE TRIGGER trigger_name
BEFORE/AFTER INSERT/UPDATE/DELETE
ON tbl_name FOR EACH ROW -- 行级触发器
BEGIN
    trigger_stmt ;
END;

        查看触发器的语法如下:

SHOW TRIGGERS ;

        删除触发器的语法如下:

-- 如果没有指定 schema_name,默认为当前数据库 。
DROP TRIGGER [schema_name.]trigger_name ; 

4.3 案例

        通过触发器记录 tb_user 表的数据变更日志,将变更日志插入到日志表 user_logs 中, 包含增加、修改和删除。

        首先创建日志存储表 user_logs,建表语句如下:

create table 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.1 插入数据触发器

create trigger tb_user_insert_trigger
after insert
on tb_user for each row
begin
	insert into 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 ;

        创建成功之后,执行一条插入语句进行测试 

insert into tb_user(id, name, phone, email, profession, age, gender, status, createtime) VALUES (25,'二皇子','18809091212','erhuangzi@163.com','软件工程',23,'1','1',now());

        可以看到,触发器被触发了,如下图 

4.3.2 更新数据触发器

create trigger tb_user_update_trigger
after update
on tb_user for each row
begin
	insert into 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;

         创建成功之后,执行一条更新语句进行测试

update tb_user set age = 32 where id = 23;

-- 由于是行级触发器,此条语句可影响多条数据,所以会存储多条数据
update tb_user set profession = '会计' where id <= 5;

        可以看到,触发器被触发了,如下图  

4.3.3 删除数据触发器

create trigger tb_user_delete_trigger
after delete
on tb_user for each row
begin
	insert into 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;

        创建成功之后,执行一条插入语句进行测试

delete from tb_user where id = 25;

        可以看到,触发器被触发了,如下图

        查看下我们创建的三个触发器,如下图:

  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

快乐的小三菊

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值