宇宙稊米 发表于 2023-5-8 21:52:30

存储过程

存储过程

介绍

存储过程是事先经过编译并存储在数据库中的一段SQL语句的集合,调用存储过程可以简化应用开发人员的很多工作,减少数据在数据库和应用服务器之间的传输,对于提高数据处理的效率是有好处的。
存储过程思想上很简单,就是数据库SQL语言层面的代码封装与重用,类似于函数或者说API,封装了系列操作,暴露接口给你进行操作。

[*]特点
封装,复用
可以接收参数,也可以返回数据
减少网络交互,效率提升
基本语法


[*]创建CREATE PROCEDURE 存储过程名称([参数列表])
BEGIN
    -- SQL语句
END;
[*]调用CALL 名称([参数]);
[*]查看-- 查询指定数据库的存储过程及状态信息
SELECT * FROM INFORMATION_SCHEMA.ROUTINES WHERE ROUTINE_SCHEMA = '数据库名称';
-- 查询某个存储过程的定义
SHOW CREATE PROCEDURE 存储过程名称;
[*]删除DROP PROCEDURE 存储过程名称;\
注意:在命令行中,执行创建存储过程的SQL语句时,需要通过关键字 delimiter 指定SQL语句的结束符。
代码演示:
-- 存储过程基本语法

-- 创建
CREATE PROCEDURE p1()
BEGIN
        SELECT 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 VARIABLES;
-- 可以通过LIKE模糊匹配方式查找变量、
SHOW VARIABLES LIKE '......';
-- 查看指定变量的值
SELECT @@ 系统变量名;
[*]设置系统变量
SET 系统变量名 = 值;
SET @@系统变量名 = 值;
注意:

[*]如果没有指定是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 变量名 变量类型;变量类型就是数据库字段类型: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()
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 p1();参数

类型含义备注IN因为该类参数作为输入,也就是需要调用时传入值默认OUT因为该类参数作为输出,也就是该类参数可以作为作为返回值INOUT既可以作为输入参数,也可以作为输出参数用法:
CREATE PROCEDURE 存储过程名称()
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)
BEGIN
        set score := score / 2;
END;

SET @score := 120;
CALL p3(@score);

SELECT @score;case


[*]语法一
CASE case_value
   WHEN when_value1 THEN statement_list1
   ...
   
END CASE;
[*]语法二
CASE
        WHEN search_condition1 THEN statement_list1
       
       
END CASE;
代码演示:
CREATE PROCEDURE p4(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 p4(4);循环-repeat
repeat是有条件的循环控制语句,直到满足条件的时候退出循环。具体语法如下:
#先判定条件,如果条件为true,则执行逻辑,否则不执行逻辑
WHILE 条件 DO
    -- SQL逻辑
END WHILE;代码演示:
#计算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
页: [1]
查看完整版本: 存储过程