存储过程

存储过程

介绍

存储过程是事先经过编译并存储在数据库中的一段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()
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 [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中配置

    代码演示:

    -- 变量:系统变量
    -- 查看系统变量
    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()
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 存储过程名称([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)
BEGIN
	set 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;
    
  • 语法二

    CASE
    	WHEN search_condition1 THEN statement_list1
    	[WHEN search_condition2 THEN statement_list2]
    	[ELSE statement_list]
    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);

循环

  • 循环-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)
BEGIN
	DECLARE 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代码的简写

原文链接:https://www.cnblogs.com/zh-Note/p/17370172.html

本站文章如无特殊说明,均为本站原创,如若转载,请注明出处:存储过程 - Python技术站

(0)
上一篇 2023年5月7日
下一篇 2023年5月9日

相关文章

  • 使用Limit参数优化MySQL查询的方法

    使用Limit参数可以在MySQL查询时控制返回的数据行数,从而优化查询效率。下面是使用Limit参数优化MySQL查询的完整攻略: 1. 什么是Limit参数 在使用SELECT语句查询数据库时,我们可以在语句的末尾使用Limit参数来限制返回的数据行数。Limit参数有两个值:第一个值指定要返回的行数(相对于结果集的第一行),第二个值可选,指定了结果集的…

    MySQL 2023年5月19日
    00
  • MySQL 1130异常,无法远程登录解决方案详解

    MySQL 1130异常,无法远程登录是MySQL在远程连接时的一个常见问题。本文详细介绍了该问题的原因以及多种解决方案。 问题原因 MySQL默认只允许localhost(127.0.0.1)上的客户端连接,如果想要从其他机器上远程连接MySQL服务器,则需要进行特殊设置。 解决方案 解决方案一:修改用户的host属性 可以通过在MySQL中为用户创建一个…

    MySQL 2023年5月18日
    00
  • MySQL关键字Distinct的详细介绍

    当我们从MySQL数据库中查询数据时,有时候会发现查询结果出现了重复的行,如果这样我们又想要保证结果唯一,这时候我们可以使用MySQL关键字DISTINCT来确保查询结果的唯一性。本文将详细介绍DISTINCT的用法和使用场景。 一、语法 MySQL中DISTINCT的语法如下所示: SELECT DISTINCT column_name(s) FROM t…

    MySQL 2023年5月19日
    00
  • MySQL 5.6主从报错的实战记录

    下面就详细讲解“MySQL 5.6主从报错的实战记录”的完整攻略。 问题描述 在 MySQL 5.6 主从复制环境中,从库报错如下: Error ‘Duplicate entry ‘12345’ for key ‘PRIMARY” on query… 问题分析 这个错误提示的含义是,由于从库上已经存在一条与主库上相同的记录,从而导致主从同步失败。查看数…

    MySQL 2023年5月18日
    00
  • Mysql性能优化案例研究-覆盖索引和SQL_NO_CACHE

    下面是关于“Mysql性能优化案例研究-覆盖索引和SQL_NO_CACHE”的详细讲解攻略。 覆盖索引 什么是覆盖索引 覆盖索引是指一个查询中的字段都可以从索引中取得,无需回表查找。这种查询方法可以提高查询效率,减少回表查询的次数,从而提高了MySQL的查询性能。 如何使用覆盖索引 具体来说,使用覆盖索引需要注意以下几点: 索引要包含查询字段和需要的返回字段…

    MySQL 2023年5月19日
    00
  • 微信昵称带符号导致插入MySQL数据库时出错的解决方案

    下面是详细讲解“微信昵称带符号导致插入MySQL数据库时出错的解决方案”的完整攻略。 问题描述 当用户在微信中设置昵称时,有可能会使用到一些特殊符号,例如“#”、“@”等。如果这些特殊符号在插入MySQL数据库时没有被转义,就有可能导致SQL语句出错,影响数据的插入或查询。下面我们来看一下具体的情况。 假设我们有一个用户表,其中包含了用户的昵称信息。我们使用…

    MySQL 2023年5月18日
    00
  • MySQL中EXPLAIN语句及用法实例

    下面是“MySQL中EXPLAIN语句及用法实例”攻略。 EXPLAIN语句在MySQL中的作用 一个查询语句,无论多么精细地编写,都可能会有性能瓶颈。常见的瓶颈有数据量太大、表太多、查询的JOIN语句过于复杂或者索引不当等。当遇到性能瓶颈问题时,我们通常需要使用MySQL的EXPLAIN语句来分析查询语句的性能瓶颈所在,从而找到最优的优化方案。 EXPLA…

    MySQL 2023年5月19日
    00
  • 配置ogg异构oracle-mysql 双向同步注意事项

    双向同步需要考虑的是怎么解决循环复制,以及同时更新一张表以谁为基准。 配置过程就不写了,大致和oracle到mysql的单向+mysql到oracle的单向差不多。 需要注意的有如下几点: 1.oracle和mysql的2端,抽取(extract)和应用(replication)应该使用不同的用户 2.为解决禁止循环复制,应该在ext进程配置3个参数,如下:…

    MySQL 2023年4月12日
    00
合作推广
合作推广
分享本页
返回顶部