MySQL存储过程、存储函数

本文详细介绍了MySQL中的存储过程和存储函数,包括它们的创建、调用、修改及删除等操作,对比了两者的差异,并探讨了它们的优点与局限。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

存储过程

Stored Procedure

一组经过预先编译的SQL语句的封装

存储过程预先在MySQL服务器上,需要执行的时候客户端只需要向服务器端发出调用存储过程的命令,服务器端就可以把预先存储好的这一系列SQL语句全部执行。

  • 好处
    1. 简化操作,提高了sql语句的重用性,减少了开发程序员的压力
    2. 减少操作过程中的失误,提高效率
    3. 减少网络传输量(客户端不需要把所有的SQL语句通过网络发给服务器)
    4. 减少了SQL语句暴露在网上的风险呢,也提高了数据查询的安全性
  • 视图与存储函数的对比
    1. 它和视图有着同样的优点,清晰、安全,还可以减少网络传输量;不同点在于,视图是虚拟表,通常不对底层数据表直接操作,而存储过程时程序化的SQL,可以直接操作底层数据表,相比于面向集合的操作方式,能够实现一些更复杂的数据处理
    2. 一旦存储过程被创建出来,使用它就像使用函数一样简单,我们直接通过存储过程名即可。相较于函数,存储过程是没有返回值的。

创建

CREATE PROCEDURE 存储过程名(IN|OUT|INOUT 形参名 参数类型,...)
[特征...]
BEGIN
	存储过程体
END

存储过程体中可以由多条SQL语句,如果仅仅一条SQL语句,则可以省略BEGIN和END编写存储过程并不是一件简单的事情,可能存储过程中需要复杂的SQL语句

BEGIN...END:BEGIN...END -- 中间包含了多个语句,每个语句都以(;)号为结束符
DECLARE:DECLARE -- 用来声明变量,使用的位置在于BEGIN...END 语句中间,而且需要在其它语句使用之前进行变量的声明
SET -- 赋值语句,用于对变量进行赋值
SELECT...INTO -- 把从数据表中查询的结果存放到变量中,也就是为变量赋值
参数分类
参数类型表现
仅仅带IN类型有参数有返回值[默认值]
仅仅带OUT类型无参数有返回值
既带IN又带OUT有参数有返回值
带INOUT有参数有返回值
特征

标识创建存储过程时指定的存储过程的约束条件

LANGUAGE SQL -- 存储过程执行体式由SQL语句组成
|[NOT] DETERMINISTIC -- 指定的结果是否正确
|{CONTAINS SQL | NO SQL |READS SQL DATA |MODIFIES SQL DATA} -- 子程序使用SQL语句的限制
|SQL SECURITY { DEFINER |INVOKER} -- 指明当前存储过程的执行权限
|COMMENT 'string'
  • LANGUAGE SQL:说明存储过程执行体式由SQL语句组成的,当前系统支持的语言为SQL

  • [NOT] DETERMINISTIC:指明存储过程指定的结果是否正确。

    DETERMINISTIC:表示结果式正确的。每次执行存储过程时,相同的输入会得到相同的输出。

    NOT DETERMINISTIC:表示结果是不确定的,相同的输入可能得到不同的输出。

    如果没有指定任意一个值,默认为NOT DETERMINISTIC

  • {CONTAINS SQL | NO SQL | READS SQL DATA | MODIFIES SQL DATA}:指明子程序使用SQL语句的限制。

    CONTAINS SQL:表示当前存储过程的子程序包含SQL语句,但是不包含读写数据的SQL语句

    NO SQL:表示当前存储过程的子程序不包含任何SQL语句

    READS SQL DATA:表示当前存储过程的子程序中包含读数据的SQL语句

    MODIFIES SQL DATA:表示当前存储过程的子程序包含写数据的SQL语句

  • SQL SECURITY { DEFINER |INVOKER}:执行当前存储过程的权限,即指明哪些用户能够执行当前存储过程

    DEFINER:表示只有当前存储过程的创建者或者定义者残念执行当前存储过程

    INVOKER:表示拥有当前存储过程的访问权限的用户能够执行当前存储过程

    如果没有设置相关的值,则MySQL默认指定值为DEFINER

  • COMMENT 'string' :注释信息,可以用来描述存储过程

结束字符
DELIMITER -- 新的结束标记
-- 因为MySQL默认的语句结束符号为';'。为了避免与存储过程中SQL语句结束符相冲突使用DELIMITER改变过程的结束符。

“DELIMITER//”语句的作用是将MySQL的结束符设置为//,并以“END//”结束存储过程。存储过程定义完毕之后使用"DELIMITER;"恢复默认结束符。DELIMITER也可以指定其它符号作为结束符

当使用DELIMITER命令时,应该避免使用反斜杠(’\‘)字符,因为反斜杠是MySQL的转义字符

DELIMITER $
CREATE PROCEDURE 存储过程名(IN|OUT|INOUT 参数名 参数类型,...)
[特征...]
BEGIN
	存储过程体
END $

调用

CALL 存储过程名(实参列表);
  • in

    CALL 存储过程名(实参列表);
    
  • out

    SET @参数;
    CALL 存储过程名(@参数);
    SELECT @参数;
    
  • inout

    SET @参数名='值';
    CALL 存储过程名(@参数名);
    SELECT @参数名;
    

举例

  • 查看emps表中所有数据

    DELIMITER $
    CREATE PROCEDURE select_all_data()
    BEGIN
    	SELECT * FROM emps;
    END$
    DELIMITER;
    
    
    CALL select_all_data();
    -- 该存储过程的调用
    
  • 查看“emps”表的最低薪资值。并将最顶薪资通过OUT参数“ms”输出

    DESC employees;
    DELIMITER $
    CREATE PROCEDURE show_min_salary(OUT ms DOUBLE)
    BEGIN
    	SELECT MIN(salary) INTO ms
    	FROM employees;
    END $
    DELIMITER;
    
    -- 调用
    CALL show_min_salary(@ms);
    -- 查看变量值
    SELECT @ms;
    
  • 查看“emps”表的某个员工的薪资,并用IN参数empname输入员工姓名

    DELIMITER $
    CREATE PROCEDURE show_someone_salary(IN empname VARCHAR(20))
    BEGIN
    	SELECT salary FROM employees
    	WHERE last_name=empname;
    END $
    DELIMITER;
    
    -- 调用
    CALL show_someone_salary('Abel');
    
    SET @empname='Abel';
    
  • 查看“emps”表的某个员工的薪资,并用IN参数empname输入员工姓名,用OUT参数empsalary输出员工的薪资。

    DELIMITER $
    CREATE PROCEDURE show_someone_salary(IN empname VARCHAR(20),OUT empsalary DECIMAL(10,2))
    BEGIN
    	SELECT salary INTO empsalary
    	FROM employees
    	WHERE last_name=empname;
    END $
    DELIMITER;
    
    -- 调用
    SET @empname;
    CALL show_someone_salary(@empname,@empsalary);
    

存储函数

CREATE FUNCTION 函数名(参数名 参数类型,...)
RETURNS 返回值类型
[特征...]
BEGIN
	函数体  -- 函数体中要求有RETURN语句
END
  1. 参数列表:指定参数为IN、OUT或INOUT只对PROCEDURE是合法的,FUNCTION中总是默认为IN参数。

  2. RETURNS type 语句表示函数返回数据的类型;

    RETURNS子句只能对FUNCTION做指定,对函数而言这是强制的。它用来指定函数的返回类型,而且函 数体必须包含一个 RETURN value 语句。

  3. characteristic 创建函数时指定的对函数的约束。取值与创建存储过程时相同,这里不再赘述。

  4. 数体也可以用BEGIN…END来表示SQL代码的开始和结束。如果函数体只有一条语句,也可以省略 BEGIN…END。

调用

在MySQL中,存储函数的使用方法与MySQL内部函数的使用方法是一样的。换言之,用户自己定义的存储函数与MySQL内部函数是一个性质的。区别在于,存储函数是用户自己定义的 ,而内部函数是MySQL的开发者定义

SELECT 函数名(实参列表);

实例

  • 创建存储函数,名称为email_by_name(),参数定义为空,改函数查询Abel的email,并返回,数据类型为字符串型

    DELIMITER $
    CREATE FUNCTION email_by_name()
    RETURNS VARCHAR(25)
            DETERMINISTIC
            CONTAINS SQL
            READS SQL DATA
    BEGIN
    	RETURN (SELECT email FROM employees WHERE last_name='Abel');
    END $
    DELIMITER;
    
    -- 调用
    SELECT email_by_name();
    
  • 参数传入emp_id,该函数查询emp_id的email,并返回,数据类型为字符串型

    SET GLOBAL log_bin_trust_function_creators=1; -- 创建函数前执行此语句,保证函数的创建会成功
    DELIMITER $
    CREATE FUNCTION email_by_id(emp_id INT)
    RETURNS VARCHAR(25)
    BEGIN
    	RETURN (SELECT email FROM employees WHERE employee_id =emp_id);
    END $
    DELIMITER;
    
    -- 调用
    SELECT email_by_id(101);
    

对比

关键字调用语法返回值应用场景
存储过程PROCEDURECALL 存储过程()0或多个更新
存储函数FUNCTIONSELECT 函数()只能一个查询结果为一个值并返回时

此外,存储函数可以凡在查询语句中使用,存储过程不行。反之,存储过程的功能更加强大,包括能够执行对表的操作(比如创建表,删除表等)和事务操作,这些功能是存储函数不具备的。

查看、修改、删除

查看

  1. 创建信息(SHOW CREATE)

    SHOW CREATE {PROCEDURE | FUNCTION}存储过程名或函数名
    
  2. 状态信息(SHOW STATUS)

    SHOW {PROCEDURE | FUNCTION [LIKE ' ']}
    
  3. 信息(information_schema.Routines)

    SELECT * FROM information_schema.Routines
    WHERE RouTINE_NAME='存储过程或函数名'
    [AND ROUTINE_TYPE={'PROCEDURE|FUNCITION'}]; -- 指明查询的是过程还是函数,(重名时用)
    

修改

修改存储过程或函数,不影响存储过程或函数,只是修改相关特性。使用ALTER语句实现

ALTER {PROCEDURE | FUNCITION}存储过程或函数的名[characteristic...]

characteristic指代功能存储过程或函数的特性,取值与创建存储过程、函数时的取值信息略有不同。

{CONTAINS SQL | NO SQL |READS SQL DATA |MODIFIES SQL DATA}
|SQL SECURITY{DEFINER |INVOKER}
|COMMENT 'sting'
  • CONTAINS SQL:表示子程序包含SQL语句,但不包含读或写数据的语句

  • NO SQL:表示子程序中不包含SQL语句

  • READS SQL DATA:表示子程序中包含读数据的语句

  • MODIFIES SQL DATA:表示子程序中包含写数据的语句

  • SQL SECURITY {DEFINER | INVOKER}:指明谁有权限来执行

    DEFINER:只有定义者自己才能够执行

    INVOKER:调用者可以执行

修改存储过程使用ALTER PROCEDURE语句,修改存储函数使用ALTER FUNCTION语句,但是,这两个语句的结构是一样的,语句汇总的所有参数也是一样的

删除

DROP {PROCEDURE | FUNCITON} [IF EXISTS]-- 存储过程或函数的名

IF EXISTS:如果程序或函数不存储,它可以防止发生错误,产生一个用SHOW WARNINGS查看的警告。

优缺点

阿里开发规范

【强制】禁止使用存储过程,存储过程难以调试和扩展,没有移植性

优点

  1. 存储过程可以依次编译多次使用
  2. 可以减少开发工作量
  3. 存储过程的安全性强
  4. 可以减少网络传输量
  5. 良好的封装性

缺点

  1. 可移植性差
  2. 调试困难
  3. 存储过程的版本管理很困难
  4. 不适合高并发的场景
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值