当前位置:网站首页>MySQL存储过程学习笔记(基于8.0)
MySQL存储过程学习笔记(基于8.0)
2022-08-04 05:31:00 【Louzen】
官方文档:https://dev.mysql.com/doc/refman/8.0/en/create-procedure.html
MySQL存储过程语法(基于8.0)
/*存储过程 & 方法 的定义:*/
CREATE
[DEFINER = user]
PROCEDURE sp_name ([proc_parameter[,...]])
[characteristic ...] routine_body
CREATE
[DEFINER = user]
FUNCTION sp_name ([func_parameter[,...]])
RETURNS type
[characteristic ...] routine_body
proc_parameter:
[ IN | OUT | INOUT ] param_name type
func_parameter:
param_name type
type:
Any valid MySQL data type
characteristic: {
COMMENT 'string'
| LANGUAGE SQL
| [NOT] DETERMINISTIC
| { CONTAINS SQL | NO SQL | READS SQL DATA | MODIFIES SQL DATA }
| SQL SECURITY { DEFINER | INVOKER }
}
routine_body:
Valid SQL routine statement
官方文档翻译
- 存储过程、方法会被定义给默认数据库,如果要指定数据库需要定义db_name.sp_name
- 可加载的方法可被视为存储过程
- 存储过程是要被调用的,在表达式中调用它,并且存储过程有返回值
- 创建存储过程或方法需要权限,如果存在DEFINER语句
- (https://dev.mysql.com/doc/refman/8.0/en/stored-objects-security.html),则权限取决于DEFINER中的user,如果需要开启二进制日志则需要超级权限
- 默认的,MySQL自动同意创建者拥有对存储过程或方法的修改、执行权限,这种默认的同意可以被系统参数automatic_sp_privileges所改变
- DEFINER和SQL安全子句指定在例程执行时检查访问权限时要使用的安全上下文(直译,不太理解)
- 尽量避免存储过程、方法名与MySQL内建的自带的方法重名,如果重名,请在定义、执行的时候在存储过程、方法名和后面的括号之间加一个空格。忽略空格的SQL语法只对MySQL内建的方法有效,而对存储过程无效,对存储过程名,无论IGNORE_SPACE参数有没有生效,都是可以在存储过程名后面加空格的
- 括号内的参数表必须始终存在,即使没有参数那参数位置也应该留有一对空括号,参数名不区分大小写
- 存储过程的参数默认是IN,若要为参数指定其他值,请在参数名称之前使用关键字OUT或INOUT,注意,只有存储过程的参数区分IN、OUT、INOUT,方法的定义中参数都是IN
- 存储过程IN参数可以在过程中改变,但存储过程的调用者不会发现这个变化(类似于java中的传值),OUT参数初始值为null,OUT值会在存储过程结束后返回给调用者,INOUT值会在传入存储过程前被调用者赋初始值,然后在存储过程执行过程中赋新值,最后执行结束把新值返回给调用者
- 因为OUT或INOUT参数会在存储过程运行结束后返回值,我们可以在其他存储过程或方法中调用一个存储过程,OUT或INOUT参数可以是外层存储过程或方法的参数、自定义的变量、表中的一列
- 若存储过程出现了未被处理的异常,则OUT和INOUT参数的值不会传回给调用方。如果异常由包含RESIGNAL语句的CONTINUE或EXIT处理程序处理,RESIGNAL的执行将弹出诊断区域堆栈,从而发出异常信号(即进入处理程序之前存在的信息)。如果异常是错误,则OUT和INOUT参数的值不会传回给调用方。
- 在例程中准备的语句中不能引用例程参数(暂时不理解,继续往后看看)
- 参数类型和函数返回类型可以声明为使用任何有效的数据类型。如果前面有字符集规范,则可以使用COLLATE属性(直译,后面一句不理解)
- 存储过程中代码体里面(BEGIN和END之间的部分)可以写简单的SQL语句比如select、insert,也可以写复合语句,复合语句可以包含生命、循环和其他控制结构语句,语法遵循:https://dev.mysql.com/doc/refman/8.0/en/sql-compound-statements.html,实际上,存储方法倾向于使用复合语句而不是只包含一个单独的return语句
- MySQL允许代码体包含DDL(数据库定义语言),比如create和drop,MySQL也允许存储过程(不是存储方法)包含SQL事务语句比如commit,存储方法不能包含执行显式或隐式提交或回滚的语句。SQL标准不要求支持这些语句,该标准规定每个DBMS供应商可以决定是否允许它们。DBMS:数据库管理软件(Data Base Management Software)或数据库管理系统(Data Base Management System)
- 存储过程中可以使用返回结果集的语句,但存储方法不可以,这种禁止包括没有INTO var_list语句的select语句,也包括show、explain、check table语句,在方法定义时写了这种语句会报“不允许返回结果集”的错误(ER_SP_NO_RETSET,https://dev.mysql.com/doc/mysql-errors/8.0/en/server-error-reference.html#error_er_sp_no_retset),而在运行时存储方法中运行了返回结果集的方法会报“在给定上下文中不能返回结果集”错误(ER_SP_BADSELECT,https://dev.mysql.com/doc/mysql-errors/8.0/en/server-error-reference.html#error_er_sp_badselect)
- 存储过程中不被允许使用use语句,当一个存储过程运行时,use db_name语句会隐式地运行(因为存储过程肯定是属于某个库的前面有说)并且在过程执行结束后撤销。使例程在执行时具有给定的默认数据库,对数据库中除例程默认数据库之外的对象的引用应使用适当的数据库名称限定(我自己的理解,存储过程默认属于某数据库,存储过程中使用默认库中的表直接用表的名字,用其他库中的表要“库.表明”这样用)
- 有关存储例程中不允许的语句的更多信息:https://dev.mysql.com/doc/refman/8.0/en/stored-program-restrictions.html
- 有关从使用具有MySQL接口的语言编写的程序中调用存储过程的信息,请参阅:https://dev.mysql.com/doc/refman/8.0/en/call.html
- MySQL在存储过程创建或修改时保存下那时的sql_mode系统变量并生效,后面运行那个存储过程时始终用这个变量去执行,无视当前服务器SQL mode是多少
- 从调用程序的SQL模式切换到例程的SQL模式发生在参数求值和将结果值分配给例程参数之后。(不理解)
- COMMENT 特性是MySQL扩展,可用于描述存储的例程,
show create procedure:https://dev.mysql.com/doc/refman/8.0/en/show-create-procedure.html;
show create function:https://dev.mysql.com/doc/refman/8.0/en/show-create-function.html - LANGUAGE 特性表示编写例程所用的语言。服务器忽略了这个特性;仅支持SQL例程
- 如果一个例程总是对相同的输入参数产生相同的结果,那么它被认为是“确定的 deterministic”,否则它就被认为是“不确定的 not deterministic”。如果例程定义中既没有给出 DETERMINISTIC 也没有给出 NOT DETERMINISTIC ,则默认为 NOT DETERMINISTIC。要声明函数是确定性的,必须显式指定 DETERMINISTIC
- 对例程性质的判定依靠创建者的“诚实”:MySQL不会检查被声明为“确定的”例程是否包含产生不确定性结果的语句。然而,错误地声明一个例程可能会影响运行结果或性能,声明一个不确定的例程为“确定 DETERMINISTIC”可能会因为使优化器选择的错误的执行计而导致未知的结果;声明一个确定的例程为“不确定 NONDETERMINISTIC”可能会因为没有使用到优化措施而降低性能
- 如果二进制日志是开启的,“确定 DETERMINISTIC”特征影响MySQL接受的例程定义(不理解)
- 包含NOW()函数(或其同义词)或RAND()的例程是不确定的,但它可能仍然是复制安全的。对于NOW()函数,二进制日志包含时间戳并且能正确拷贝;RAND()也可以正确复制,只要在例程中函数只调用了一次。我们可以看做在复制时,在源和复制品中执行这两个方法时用了相同的时间戳和随机数生成种子,这样就能使通过二进制文件拷贝的复制品和源的数据一致
- 几个特性提供了例程所用的数据的性质的信息(不理解),在MySQL中这些特性是建议性的,服务器不会用它们约束例程允许执行哪种语句。
- CONTAINS SQL 表明例程没有包含读写数据的语句,如果这些特性没有明确给出那这就是默认值。比如有语句:"SET @X = 1"或"DO RELEASE_LOCK(‘abc’)"这些语句执行起来都没有读写数据
- NO SQL 表明例程中没有SQL语句
- READS SQL DATA 表明例程包含读数据的语句(比如select)但是没有语句用来写数据
- MODIFIES SQL DATA 表明例程包含可能写数据的语句(比如insert或delete)
- SQL SECURITY 特性能被DEFINER定义者或INVOKER调用者指定安全上下文;就是,用谁的账号的权限执行例程,是例程中DEFINER语句中定义命名的用户?还是执行例程的用户?这个账户必须有使用例程所属的数据库的权限,SQL SECURITY默认值是DEFINER,即由DEFINER语句中定义命名的用户的账号的权限去执行例程,调用例程的用户必须具有该例程的执行EXECUTE权限,如果例程在定义者安全上下文中执行,则定义者帐户也必须具有该权限(不理解,什么是“安全上下文”?自我理解,安全上下文就是用谁的权限来执行存储过程中的代码内容。首先,调用者需要又调用存储过程的权限,而存储过程执行过程中,对哪些库、哪些表执行什么样的操作,以及其他非对库、表的操作,做这些操作的权限,是要看SQL SECURITY 后面指定DEFINER还是INVOKER,其中SQL SECURITY DEFINER是默认值,可以不用写,如果需要指定INVOKER需要显式地写出来)
- DEFINER子句指定在例程执行时检查具有SQL SECURITY DEFINER特性的例程的访问权限时要使用的MySQL帐户
- 如果有DEFINER子句,那user字段的格式应该是 ‘user_name’@‘host_name’,CURRENT_USER或CURRENT_USER()。允许的user值取决于我拥有的权限,有关存储过程安全的更多内容看:https://dev.mysql.com/doc/refman/8.0/en/stored-objects-security.html
- 如果DEFINER特性被省略了,默认的定义者就是执行存储过程或方法的用户,这与写“DEFINER = CURRENT_USER”效果一样
- 在定义了SQL SECURITY DEFINER的例程中的代码体中,CURRENT_USER方法返回的就是例程的DEFINER值,对于存储例程中的用户审核,看:https://dev.mysql.com/doc/refman/8.0/en/account-activity-auditing.html
- 看下面“DEFINER ‘admin’@‘localhost’”代码块:不管是哪个用户定义的,存储过程中都会定义’admin’@'localhost’用户为DEGINER;不管是哪个用户执行存储过程,存储过程的执行都会使用DEFINER指定的用户的权限。存储过程执行的成败取决于调用者是否具备执行权限、DEFINER指定的admin用户是否有对mysql.user表的查询权限
- 看下面“SQL SECURITY INVOKER”代码块,现在指定安全上下文为存储过程调用者SQL SECURITY INVOKER:存储过程中仍指定DEFINER = ‘admin’@‘localhost’,但是在这个例子中,存储过程是用调用者的权限执行的,因此,存储过程执行成败取决于调用者是否有存储过程的调用权限和对mysql.user表的select权限
- 服务器处理存储过程参数、存储过程方法内用DECLARE创建的本地变量、方法返回值等的数据类型:
- 为数据类型不匹配和溢出问题而进行匹配检查,在严格的SQL模式下转换和溢出问题会导致警告、错误
- 只能指定标量值,比如,“set x = (select 1,2)”是非法的
- 对于字符数据类型,如果CHARACTER SET被包含进声明中时,指定的字符集和它默认校验规则会被使用,如果COLLATE参数也被声明,则校验就会用COLLATE指定的校验规则而不是默认的
如果CHARACTER SET和COLLATE都没有声明,使用在例程创建时有效的数据库字符集和排序规则。为了避免让服务器使用数据库字符集和校验规则,就提供了显式的字符数据参数的设置CHARACTER SET和COLLATE
如果你改变了数据库默认字符集和校验规则,使用新数据库默认值的存储过程一定要要被删除并重新创建
数据库字符集和校验规则被character_set_database和collation_database两个系统变量,更多信息看:https://dev.mysql.com/doc/refman/8.0/en/charset-database.html
DEFINER = ‘admin’@‘localhost’
CREATE DEFINER = 'admin'@'localhost' PROCEDURE account_count()
BEGIN
SELECT 'Number of accounts:', COUNT(*) FROM mysql.user;
END;
SQL SECURITY INVOKER
CREATE DEFINER = 'admin'@'localhost' PROCEDURE account_count()
SQL SECURITY INVOKER
BEGIN
SELECT 'Number of accounts:', COUNT(*) FROM mysql.user;
END;
边栏推荐
- 【Copy攻城狮日志】飞浆学院强化学习7日打卡营-学习笔记
- 语音驱动嘴型与面部动画生成的现状和趋势
- arm learning-1-development board
- MNIST Handwritten Digit Recognition - Image Analysis Method for Binary Classification
- arm-3-中断体系结构
- Amazon Cloud Technology Build On-Amazon Neptune's Knowledge Graph-Based Recommendation Model Building Experience
- 亚马逊云科技Build On-Amazon Neptune基于知识图谱的推荐模型构建心得
- 集合---ArrayList的底层
- FAREWARE ADDRESS
- Code to celebrate the Dragon Boat Festival - Zongzi, your heart
猜你喜欢
打金?工作室?账号被封?游戏灰黑产离我们有多近
MNIST Handwritten Digit Recognition - Building a Perceptron from Zero for Two-Classification
【论文阅读】Exploring Spatial Significance via Hybrid Pyramidal Graph Network for Vehicle Re-identificatio
Copy攻城狮5分钟在线体验 MindIR 格式模型生成
arm-2-基础阶段
No matching function for call to 'RCTBridgeModuleNameForClass'
深度学习理论——过拟合、欠拟合、正则化、优化器
亚马逊云科技 Build On 2022 - AIot 第二季物联网专场实验心得
基于asp.net的法律援助平台的设计与实现(附项目链接)
【Copy攻城狮日志】飞浆学院强化学习7日打卡营-学习笔记
随机推荐
剪映专业版字幕导出随笔
LeetCode_Dec_1st_Week
学习资料re-id
Pytest常用插件
[开发杂项][VS Code]remote-ssd retry failed
第一章 绪论
MNIST手写数字识别 —— 基于Mindspore快速构建感知机实现十分类
Pytest common plug-in
双向LSTM
MVC自定义配置
Question 1000: Input two integers a and b, calculate the sum of a+b, this question is multiple sets of test data
深度确定性策略梯度(DDPG)
安装Apache服务时出现的几个问题, AH00369,AH00526,AH00072....
数据库的简述与常用操作指南
Amazon Cloud Technology Build On-Amazon Neptune's Knowledge Graph-Based Recommendation Model Building Experience
Introduction to Convolutional Neural Networks
如何用Pygame制作简单的贪吃蛇游戏
彻底删除MySQL教程
FAREWARE ADDRESS
MNIST Handwritten Digit Recognition - Lenet-5's First Commercial Grade Convolutional Neural Network