存储过程介绍
存储过程是事先经过编译并存储在数据库中的一段SQL语句的集合,调用存储过程可以简化应用开发人员的很多工作,减少数据在数据库和应用服务器之间的传输,对于提高数据处理的效率是有好处的。
存储过程思想上很简单,就是数据库SQL语言层面的代码封装与重用,类似于函数或者说API,封装了系列操作,暴露接口给你进行操作。
- 特点
封装,复用
可以接收参数,也可以返回数据
减少网络交互,效率提升
基本语法
- 创建
CREATE PROCEDURE 存储过程名称([参数列表])BEGIN -- SQL语句END;
- 调用
CALL 名称([参数]);
- 查看
-- 查询指定数据库的存储过程及状态信息SELECT * FROM INFORMATION_SCHEMA.ROUTINES WHERE ROUTINE_SCHEMA = '数据库名称';-- 查询某个存储过程的定义SHOW CREATE PROCEDURE 存储过程名称;
- 删除
DROP PROCEDURE [IF EXISTS] 存储过程名称;\
注意:在命令行中,执行创建存储过程的SQL语句时,需要通过关键字 delimiter 指定SQL语句的结束符。
代码演示:
-- 存储过程基本语法-- 创建CREATE PROCEDURE p1()BEGINSELECT COUNT(*) FROM emp;END;-- 如果在命令行执行的话,上面语句会出错,因为遇到第一个分号就结束了,-- 所以需要用到 delimiter-- delimiter $$ --表示指定 $$ 为结束符(记得之后再使用delimiter改回来)-- CREATE PROCEDURE p1()-- BEGIN-- SELECT COUNT(*) FROM emp;-- END;$$-- 这样就可以解决-- 调用CALL p1();-- 查看SELECT * FROM INFORMATION_SCHEMA.ROUTINES WHERE ROUTINE_SCHEMA = 'hsp_db02';SHOW CREATE PROCEDURE p1;-- 删除DROP PROCEDURE IF EXISTS p1;
变量
系统变量
系统变量是MySQL服务器提供,不是用户定义的,属于服务器层面。分为全局变量(GLOBAL)、会话变量(SESSION)。查看系统变量
-- 查看所有系统变量SHOW [SESSION|GLOBAL] VARIABLES;-- 可以通过LIKE模糊匹配方式查找变量、SHOW [SESSION|GLOBAL] VARIABLES LIKE '......';-- 查看指定变量的值SELECT @@[SESSION|GLOBAL] 系统变量名;
设置系统变量
SET [SESSION|GLOBAL] 系统变量名 = 值;SET @@[SESSION|GLOBAL]系统变量名 = 值;
注意:
- 如果没有指定是session或global,则默认是session,会话变量。
- mysql服务重新启动后,所设置的全局参数会失效,要想不失效,可以在/etc/my.cnf中配置
代码演示:
-- 变量:系统变量-- 查看系统变量show variables; -- 所有系统变量show session variables; -- 会话变量show global variables; -- 全局变量show session variables like 'auto%';show global variables like 'auto%';select @@global.autocommit;select @@session.autocommit;-- 设置系统变量set session autocommit = 0; -- 设置为0,代表关闭了当前会话的事务的自动提交set global autocommit = 0; -- 当服务器重启后,这个参数又会初始化为默认值,想要不失效,需要在etc/my.cnf中配置
用户定义变量
用户定义变量 是用户根据需要自己定义的变量,用户不用提前声明,在用的时候直接用 “@变量名”使用就可以。其作用域为当前连接(会话)。赋值
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;
注意:用户定义的变量是无需对其进行声明或初始化的,只不过获取到的值为NULL。
代码演示:-- 变量:用户变量-- 赋值SET @myname = 'itcast';SET @myage := 10; -- 推荐set @mygender := '男', @myhobby := 'java';SELECT @mycolor := 'red';SELECT COUNT(*) into @mycount FROM demo;-- 使用SELECT @myname, @myage, @mygender, @myhobby;SELECT @mycolor, @mycount;SELECT @abc; -- NULL
局部变量
局部变量是根据需要定义在局部生效的变量,访问之前,需要DECLARE声明。可以用作存储过程内的局部变量和输入参数,局部变量的范围是在其内声明的BEGIN … END块中。声明
DECLARE 变量名 变量类型[DEFAULT ...];
变量类型就是数据库字段类型:INT、BiGINT、CHAR、VARCHAR、DATE、TIME等。
赋值
SET 变量名 = 值;SET 变量名 := 值;SELECT 字段名 INTO 变量名 FROM 表名...;
代码演示:
-- 变量:局部变量-- 声明 declare-- 赋值CREATE PROCEDURE p2()BEGIN DECLARE stu_count INT DEFAULT 0; set stu_count := 100; SELECT COUNT(*) into stu_count FROM demo; SELECT stu_count;END;CALL p2();
if判断
语法:
IF 条件1 THEN......ELSEIF 条件2 THEN -- 可选......ELSE -- 可选......END IF;
代码演示:
CREATE PROCEDURE p1()BEGINDECLARE 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 p1();
参数
类型 | 含义 | 备注 |
---|---|---|
IN | 因为该类参数作为输入,也就是需要调用时传入值 | 默认 |
OUT | 因为该类参数作为输出,也就是该类参数可以作为作为返回值 | |
INOUT | 既可以作为输入参数,也可以作为输出参数 |
用法:
CREATE PROCEDURE 存储过程名称([IN/OUT/INOUT 参数名 参数类型])BEGIN -- SQL语句END;
代码演示:
CREATE PROCEDURE p2(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;CALL p2(68, @result);SELECT @result;CREATE PROCEDURE p3(INOUT score INT)BEGINset score := score / 2;END;SET @score := 120;CALL p3(@score);SELECT @score;
case
语法一
CASE case_value WHEN when_value1 THEN statement_list1 [WHEN when_value2 THEN statement_list2]... [ELSE statement_list]END CASE;
语法二
CASEWHEN search_condition1 THEN statement_list1[WHEN search_condition2 THEN statement_list2][ELSE statement_list]END CASE;
代码演示:
CREATE PROCEDURE p4(in month int)BEGINdeclare result varchar(10); case when month >= 1 and month = 4 and month = 7 and month = 10 and month <= 12 THEN set result := '第四季度'; else SET result := '非法参数'; END case;SELECT concat('您输入的月份为:', month, ' 所属的季度为:', result);END;CALL p4(4);
循环
循环-while
while循环是有条件的循环控制语句,满足条件后在执行循环体内的SQL语句,具体语法如下:#先判定条件,如果条件为true,则执行逻辑,否则不执行逻辑WHILE 条件 DO -- SQL逻辑END WHILE;
代码演示:
#计算1到n的累加值CREATE procedure p5(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 p5(10);
循环-repeat
repeat是有条件的循环控制语句,直到满足条件的时候退出循环。具体语法如下:#先执行一次逻辑,然后判定是否满足条件,如果满足则退出,不满足,继续进行下一次循环REPEAT -- SQL逻辑 -- UNTIL 条件END REPEAT;
代码演示:
#计算1到n的累加值CREATE procedure p6(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 p6(10);
循环-loop
LOOP实现简单的循环,如果不在SQL逻辑中增加退出循环的条件,可以用其来实现简单的死循环。LOOP可以配合以下两个语句使用:- LEAVE:配合循环使用,退出循环。
- ITERATE:必须用在循环中,作用是跳过当前循环剩下的语句,直接进入下一次循环。
[begin_label:] LOOP -- SQL逻辑END LOOP [end_label];
LEAVE label; -- 退出指定标记的循环体ITERATE label; -- 直接进入下一次循环
代码演示:
#计算1到n的累加值CREATE procedure p7(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 p7(10);#计算1到n中的偶数的累加值CREATE procedure p8(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 p8(10);
游标
游标(CURSOR)是用来存储查询结果集的数据类型,在存储过程和函数中可以使用游标对结果集进行循环的处理。游标的使用包括游标的声明、OPEN、FETCH 和 CLOSE,其语法分别如下:
- 声明游标
DECLARE 游标名称 CURSOR FOR 查询语句;
- 打开游标
OPEN 游标名称;
- 获取游标记录
FETCH 游标名称 INTO 变量[,变量];
- 关闭游标
CLOSE 游标名称
CREATE procedure p1(in uage int)BEGINDECLARE u_name varchar(32); DECLARE u_profession varchar(32); DECLARE u_cursor CURSOR for SELECT name, profession FROM `user` where age <= uage; -- DECLARE exit handler for sqlstate '02000' CLOSE u_cursor; DECLARE exit handler for Not FOUND CLOSE u_cursor; DROP TABLE if exists user_pro; CREATE TABLE user_pro( id int PRIMARY key AUTO_INCREMENT, name varchar(32), profession varchar(32) ); OPEN u_cursor; WHILE true do FETCH u_cursor INTO u_name, u_profession; INSERT into user_pro VALUES(null, u_name, u_profession); end WHILE;end;CALL p1(25);SELECT * from user_pro;
user表:
id | name | age | profession |
---|---|---|---|
1 | 孙悟空 | 20 | java |
2 | 猪八戒 | 21 | php |
3 | 沙和尚 | 22 | C |
4 | 唐僧 | 19 | C++ |
5 | 观音 | 24 | C# |
6 | 佛祖 | 30 | js |
7 | 玉皇大帝 | 31 | ts |
8 | 西王母 | 29 | python |
9 | 哪吒 | 15 | go |
user_pro表:
id | name | profession |
---|---|---|
1 | 孙悟空 | java |
2 | 猪八戒 | php |
3 | 沙和尚 | C |
4 | 唐僧 | C++ |
5 | 观音 | C# |
6 | 哪吒 | go |
条件处理程序-handler
条件处理程序(Handler)可以用来定义在流程控制结构执行过程中遇到问题时相应的处理步骤,具体语法如下:
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代码的简写