五、MYSQL存储过程和函数

五、MYSQL存储过程和函数,第1张

• create procedure用来创建 存储过程 ,create function用来创建 函数

Delimiter命令是改变语句的结束符 ,MySQL默认的结束符为号,由于procedure和function中的号并不代表创建的结束,所以要替换成另外的结束符以便表示创建的结束

• rontine_body子句可以包含一个简单的SQL语句,也可以包含多个SQL语句, 通过begin…end将这多个SQL语句 包含在一起

• MySQL存储过程和函数中也可以包含类似create和drop等DDL语句

• comment子句用来写入对存储过程和函数的注释

Language子句用来表示此存储过程和函数的创建语言

存储过程和函数被标注为deterministic表明当输入相同的参数是会返回相同的结果,反之如果是not deterministic则表示相同参数不会是相同结果,默认是not deterministic

相关属性短语只有咨询含义,并不是强制性的约束

• Drop procedure/function语句用来 删除指定名称的存储过程或函数

• Begin…end语句通常出现在存储过程、函数和触发器中,其中 可以包含一个或多个语句 ,每个语句用号隔开

• 标签label可以加在begin…end语句以及loop, repeat和while语句

语句中通过iterate和leave来控制流程,iterate表示返回指定标签位置,leave表示跳出标签

Declare语句通常用来声明本地变量、游标、条件或者handler

Declare语句只允许出现在begin … end语句中而且必须出现在第一行

Declare的顺序也有要求,通常是先声明本地变量,再是游标,然后是条件和handler

• 本地变量可以通过declare语句进行声明

声明后的变量可以通过select … into var_list进行赋值,或者通过set语句赋值,或者通过定义游标并使用fetch … into var_list赋值

• 通过declare声明变量方法:

• MySQL支持if,case,iterate,leave,loop,while,repeat语句作为存储过程和函数中的 流程控制语句 ,另外return语句也是函数中的特定流程控制语句

• Case语句在存储过程或函数中表明了 复杂的条件选择语句

• IF语句在存储过程或函数中表明了 基础的条件选择语句

其中在 function 里面,只有 DETERMINISTIC, NO SQL 和 READS SQL DATA 被支持。如果我们开启了 bin-log, 我们就必须为我们的 function 指定一个参数。

在 MySQL 中创建函数时出现这种错误的解决方法:

set global log_bin_trust_function_creators=TRUE

• Iterate语句 仅出现在loop,repeat,while循环语句中,其含义表示重新开始此循环

• Leave语句表明 退出指定标签的流程控制语句块

• 通常会用在begin…end,以及loop,repeat,while的循环语句中

• Loop语句是存储过程或函数中表达 循环执行 的一种方式

• repeat语句是存储过程或函数中表达 循环执行 的一种方式

• while语句是存储过程或函数中表达 循环执行 的一种方式

• Return语句用在 函数中,用来终结函数的执行并将指定值返回给调用者

• Cursor游标用来 声明一个数据集

• 游标的声明必须在变量和条件声明之后,在handler声明之前

• Cursor close语句用来 关闭之前打开的游标

• Cursor declare语句用来声明一个游标和指定游标对应的数据集合, 通常数据集合是一个select语句

• Cursor fetch语句用来获取游标指定数据集的 下一行数据 并将各个字段值赋予后面的变量

• Open cursor语句用来打开一个之前已经 声明好的游标

• Declare condition语句命名 特定的错误条件 ,而该特定错误可以在declare…handler中指定 处理方法

• 比如在MySQL中1051error code表示的是unknown table的错误,如果要对这

个错误做特殊处理,可以用三种方法:

• Declare handler语句用来声明一个handler来处理一个或多个特殊条件,当其中的某个条件满足时则触发其中的statement语句执行

• Statement可以是一个简单SQL语句,也可以是begin…end组成的多个语句

• Handler_action子句声明当执行完statement语句之后应该怎么办

Condition_value的值有以下几种:

• 当condition发生但没有声明handler时,则存储过程和函数依照如下规则处理

• create trigger语句用来创建一个触发器,触发器的作用是当表上有对应SQL语句发生时,则触发执行

• 触发器创建时需要 指定对应的表名 tbl_name

Definer关键词用来指定trigger的安全环境

• Trigger_time指定触发器的执行时间,BEFORE和AFTER指定触发器在表中的 每行数据修改前或者后 执行

• Trigger_event指定触发该触发器的具体 事件

• INSERT当新的一行数据插入表中时触发,比如通过执行insert,load data,replace语句插入新数据

• UPDATE当表的一行数据被修改时触发,比如执行update语句时

• DELETE当表的一行数据被删除时触发,比如执行delete,replace语句时

• 当执行insert into … on duplicate key update语句时,当碰到重复行执行update时,则触发update下的触发器

• 从5.7.2版本开始,可以创建具有相同trigger_time和trigger_event的同一个表上的多个触发器,默认情况下按照创建的时间依次执行,通过 指定FOLLOWS/PRECEDES改变执行顺序 ,即FOLLOWS时表示新创建的触发器后执行,PRECEDES则表示新触发器先执行

• Trigger_body表示触发器触发之后要执行的一个或多个语句,在内部可以引用涉及表的字段, OLD.col_name表示行数据被修改或删除之前的字段数据,NEW.col_name表示行数据被插入或修改之后的字段数据

• Drop trigger语句用来 删除一个触发器

• If exists短语用来避免删除不存在的触发器时引发报错

当你执行drop table时,表上的触发器也被drop掉了

有两种方法,一种方法使用mysql的check table和repair table 的sql语句,另一种方法是使用MySQL提供的多个myisamchk, isamchk数据检测恢复工具。前者使用起来比较简便。推荐使用。

1. check table 和 repair table

登陆mysql 终端:

mysql -uxxxxx -p dbname

check table tabTest

如果出现的结果说Status是OK,则不用修复,如果有Error,可以用:

repair table tabTest

进行修复,修复之后可以在用check table命令来进行检查。在新版本的phpMyAdmin里面也可以使用check/repair的功能。

2. myisamchk, isamchk

其中myisamchk适用于MYISAM类型的数据表,而isamchk适用于ISAM类型的数据表。这两条命令的主要参数相同,一般新的系统都使用MYISAM作为缺省的数据表类型,这里以myisamchk为例子进行说明。当发现某个数据表出现问题时可以使用:

myisamchk tablename.MYI

进行检测,如果需要修复的话,可以使用:

myisamchk -of tablename.MYI

关于myisamchk的详细参数说明,可以参见它的使用帮助。需要注意的时在进行修改时必须确保MySQL服务器没有访问这个数据表,保险的情况下是最好在进行检测时把MySQL服务器Shutdown掉。

另外可以把下面的命令放在你的rc.local里面启动MySQL服务器前:

[ -x /tmp/mysql.sock ] &&/pathtochk/myisamchk -of /DATA_DIR/*/*.MYI

其中的/tmp/mysql.sock是MySQL监听的Sock文件位置,对于使用RPM安装的用户应该是/var/lib/mysql/mysql.sock,对于使用源码安装则是/tmp/mysql.sock可以根据自己的实际情况进行变更,而pathtochk则是myisamchk所在的位置,DATA_DIR是你的MySQL数据库存放的位置。

需要注意的时,如果你打算把这条命令放在你的rc.local里面,必须确认在执行这条指令时MySQL服务器必须没有启动!检测修复所有数据库(表)

测试表与测试数据

CREATE TABLE TestMulToOne (

  name       VARCHAR(10),

  allTitles  VARCHAR(200)

)

INSERT INTO TestMulToOne VALUES ('张三', '程序员,系统管理员,网络管理员')

INSERT INTO TestMulToOne VALUES ('李四', '项目经理,系统分析员')

   

 

DELIMITER //

DROP FUNCTION GetTextCount //

/**********

-- 获取字符串中有几个部分.

**********/

CREATE FUNCTION GetTextCount(pSourceText  VARCHAR(255),  pDivChar  CHAR(1))

RETURNS TINYINT

BEGIN

 -- 预期结果. 

 DECLARE vResult TINYINT

 -- 当前逗号的位置.

 DECLARE vIndex INT

 -- 前一个逗号的位置.

 DECLARE vPrevIndex INT

 -- 结果的初始值.

 SET vResult = 1

 -- 查询第一个 逗号的位置.

 SET vIndex = INSTR(pSourceText, pDivChar)

 IF vIndex = 0 THEN

  -- 参数中没有逗号,直接返回.

  RETURN vResult

 END IF

 -- 初始化情况,前一个逗号不存在.

 SET vPrevIndex = 0

 -- 循环处理。

 WHILE vIndex > 0 DO

  -- 结果递增.

  SET vResult = vResult + 1  

  -- 前一个逗号的位置 = 当前逗号的位置

  SET vPrevIndex = vIndex

  -- 查询下一个逗号的位置.

  SET vIndex = LOCATE(pDivChar,  pSourceText,  vPrevIndex + 1)

 END WHILE

 -- 返回结果.

 RETURN vResult

END

//

 

DROP FUNCTION GetTextValue //

/**********

-- 获取字符串中具体某一个部分的数据.

**********/

CREATE FUNCTION GetTextValue(pSourceText  VARCHAR(255),  pDivChar  CHAR(1), pIndex  TINYINT)

RETURNS VARCHAR(255) 

BEGIN

 -- 预期结果. 

 DECLARE vResult VARCHAR(255)

 IF pIndex = 1 THEN

  SELECT SUBSTRING_INDEX(pSourceText,  pDivChar,  1)  INTO  vResult

 ELSE

  SELECT 

   REPLACE(

    SUBSTRING_INDEX(pSourceText,  pDivChar,  pIndex),

    CONCAT(SUBSTRING_INDEX(pSourceText,  pDivChar,  pIndex - 1) , pDivChar),

    '') INTO  vResult

 END IF

 

 -- 返回.

 RETURN vResult

END

//

DELIMITER 

/*

SELECT GetTextCount('程序员,系统管理员,网络管理员', ',') AS Co

SELECT 

 GetTextValue('程序员,系统管理员,网络管理员', ',', 1) AS A,

 GetTextValue('程序员,系统管理员,网络管理员', ',', 2) AS B,

 GetTextValue('程序员,系统管理员,网络管理员', ',', 3) AS C

*/

   

SELECT

 t.name,

 GetTextValue(t.allTitles, ',', MaxNum.No) AS OneTitle

FROM

 TestMulToOne t,

 (SELECT 1 No UNION ALL

  SELECT 2 No UNION ALL

  SELECT 3 No UNION ALL

  SELECT 4 No UNION ALL

  SELECT 5 No ) MaxNum

WHERE

 GetTextCount(t.allTitles, ',') >= MaxNum.No

ORDER BY

 t.name

 

+------+------------+

| name | OneTitle   |

+------+------------+

| 张三 | 系统管理员 |

| 张三 | 网络管理员 |

| 张三 | 程序员     |

| 李四 | 系统分析员 |

| 李四 | 项目经理   |

+------+------------+

5 rows in set (0.00 sec)


欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/zaji/8389696.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2023-04-16
下一篇 2023-04-16

发表评论

登录后才能评论

评论列表(0条)

保存