MySQL_变量、流程控制与游标

变量


在MySQL数据库的存储过程和函数中,可以使用变量来存储查询或计算的中间结果数据,或者输出最终的结果数据。

 在 MySQL 数据库中,变量分为 系统变量 以及 用户自定义变量

系统变量

变量由系统定义,不是用户定义,属于服务器层面。启动MySQL服务,生成MySQL服务实例期间, MySQL将为MySQL服务器内存中的系统变量赋值,这些系统变量定义了当前MySQL服务实例的属性、特征。这些系统变量的值要么是 编译MySQL时参数 的默认值,要么是配置文件(例如my.ini等中的参数值

系统变量分为全局系统变量(需要添加 global 关键字)以及会话系统变量(需要添加 session 关键 字),有时也把全局系统变量简称为全局变量,有时也把会话系统变量称为local变量。如果不写,默认为会话级别。静态变量(在 MySQL 服务实例运行期间它们的值不能使用 set 动态修改)属于特殊的全局系统变量。

每一个MySQL客户机成功连接MySQL服务器后,都会产生与之对应的会话。会话期间,MySQL服务实例 会在MySQL服务器内存中生成与该会话对应的会话系统变量,这些会话系统变量的初始值是全局系统变 量值的复制。

全局系统变量针对于所有会话(连接)有效,但不能跨重启

在MySQL中有些系统变量只能是全局的例如 max_connections 用于限制服务器的最大连接数;

有些系统变量作用域既可以是全局又可以是会话,例如 character_set_client 用于设置客户端的字符集;

有些系统变量的作用域只能是当前会话,例如 pseudo_thread_id 用于标记当前会话的 MySQL 连接 ID。

查看系统变量

#查看所有全局变量

SHOW GLOBAL VARIABLES;

#查看所有会话变量

SHOW SESSION VARIABLES;

SHOW VARIABLES;

#查看满足条件的部分系统变量。

SHOW GLOBAL VARIABLES LIKE '%标识符%';

#查看满足条件的部分会话变量

SHOW SESSION VARIABLES LIKE '%标识符%';

举例:

SHOW GLOBAL VARIABLES LIKE 'admin_%'; 

查看指定系统变量

作为 MySQL 编码规范,MySQL 中的系统变量以 两个“@” 开头,其中“@@global”仅用于标记全局系统变 量,“@@session”仅用于标记会话系统变量。“@@”首先标记会话系统变量,如果会话系统变量不存在, 则标记全局系统变量。

#查看指定的系统变量的值

SELECT @@global.变量名;

#查看指定的会话变量的值

SELECT @@session.变量名;

#或者

SELECT @@变量名; 

修改系统变量的值

有些时候,数据库管理员需要修改系统变量的默认值,以便修改当前会话或者MySQL服务实例的属性、 特征。具体方法:

方式1:修改MySQL 配置文件 ,继而修改MySQL系统变量的值(该方法需要重启MySQL服务) 方式2:在MySQL服务运行期间,使用“set”命令重新设置系统变量的值 

#方式1:

SET @@global.变量名=变量值;

#方式2:

SET GLOBAL 变量名=变量值; #为某个会话变量赋值

#方式1:

SET @@session.变量名=变量值;

#方式2:

SET SESSION 变量名=变量值;

用户变量 

用户变量是用户自己定义的,作为 MySQL 编码规范,MySQL 中的用户变量以 一个“@” 开头。根据作用 范围不同,又分为 会话用户变量 和 局部变量 。

  • 会话用户变量:作用域和会话变量一样,只对 当前连接 会话有效。
  • 局部变量:只在 BEGIN 和 END 语句块中有效。局部变量只能在 存储过程和函数 中使用。

会话用户变量 

变量的定义

#方式1:“=”或“:=”

SET @用户变量 = 值;

SET @用户变量 := 值;

#方式2:“:=” 或 INTO关键字

SELECT @用户变量 := 表达式 [FROM 等子句];

SELECT 表达式 INTO @用户变量 [FROM 等子句];

查看用户变量的值 

SELECT @用户变量

局部变量 

定义:可以使用 DECLARE 语句定义一个局部变量

作用域:仅仅在定义它的 BEGIN ... END 中有效

位置:只能放在 BEGIN ... END 中,而且只能放在第一句

BEGIN

        #声明局部变量

        DECLARE 变量名1 变量数据类型 [DEFAULT 变量默认值];

        DECLARE 变量名2,变量名3,... 变量数据类型 [DEFAULT 变量默认值];

        #为局部变量赋值

        SET 变量名1 = 值;

        SELECT 值 INTO 变量名2 [FROM 子句];

        #查看局部变量的值

        SELECT 变量1,变量2,变量3;

END 

1.定义变量

DECLARE 变量名 类型 [default 值]; # 如果没有DEFAULT子句,初始值为NULL 

2.变量赋值

用于赋简单的值

SET 变量名=值;

SET 变量名:=值; 

用于赋表中的字段值

SELECT 字段名或表达式 INTO 变量名 FROM 表; 

流程控制


解决复杂问题不可能通过一个 SQL 语句完成,我们需要执行多个 SQL 操作。流程控制语句的作用就是控制存储过程中 SQL 语句的执行顺序,是我们完成复杂操作必不可少的一部分。只要是执行的程序,流程 就分为三大类:

  • 顺序结构 :程序从上往下依次执行
  • 分支结构 :程序按条件进行选择执行,从两条或多条路径中选择一条执行
  • 循环结构 :程序满足一定条件下,重复执行一组语句

 针对于MySQL 的流程控制语句主要有 3 类。注意:只能用于存储程序。

  • 条件判断语句 :IF 语句和 CASE 语句
  • 循环语句 :LOOP、WHILE 和 REPEAT 语句
  • 跳转语句 :ITERATE 和 LEAVE 语句

 分支结构——IF

 IF 语句的语法结构是:

IF 表达式1 THEN 操作1

[ELSEIF 表达式2 THEN 操作2]……

[ELSE 操作N]

END IF

 根据表达式的结果为TRUE或FALSE执行相应的语句。这里“[]”中的内容是可选的。

特点:① 不同的表达式对应不同的操作 ② 使用在begin end中

分支结构——CASE

 CASE 语句的语法结构1:

#情况一:类似于switch

CASE 表达式

WHEN 值1 THEN 结果1或语句1(如果是语句,需要加分号)

WHEN 值2 THEN 结果2或语句2(如果是语句,需要加分号)

...

ELSE 结果n或语句n(如果是语句,需要加分号)

END [case](如果是放在begin end中需要加上case,如果放在select后面不需要)

CASE 语句的语法结构2:

#情况二:类似于多重if

CASE

WHEN 条件1 THEN 结果1或语句1(如果是语句,需要加分号)

WHEN 条件2 THEN 结果2或语句2(如果是语句,需要加分号)

...

ELSE 结果n或语句n(如果是语句,需要加分号)

END [case](如果是放在begin end中需要加上case,如果放在select后面不需要)

循环结构——LOOP 

LOOP循环语句用来重复执行某些语句。LOOP内的语句一直重复执行直到循环被退出(使用LEAVE子 句),跳出循环过程。

LOOP语句的基本格式如下:

[loop_label:] LOOP

    循环执行的语句

END LOOP [loop_label]

 其中,loop_label表示LOOP语句的标注名称,该参数可以省略

循环结构——WHILE

WHILE语句创建一个带条件判断的循环过程。WHILE在执行语句执行时,先对指定的表达式进行判断,如 果为真,就执行循环内的语句,否则退出循环。WHILE语句的基本格式如下:

[while_label:] WHILE 循环条件 DO

        循环体

END WHILE [while_label];

 while_label为WHILE语句的标注名称;如果循环条件结果为真,WHILE语句内的语句或语句群被执行,直 至循环条件为假,退出循环。 

游标


什么是游标(光标)

虽然我们也可以通过筛选条件 WHERE 和 HAVING,或者是限定返回记录的关键字 LIMIT 返回一条记录, 但是,却无法在结果集中像指针一样,向前定位一条记录、向后定位一条记录,或者是 随意定位到某一 条记录 ,并对记录的数据进行处理。

这个时候,就可以用到游标。游标,提供了一种灵活的操作方式,让我们能够对结果集中的每一条记录 进行定位,并对指向的记录中的数据进行操作的数据结构。游标让 SQL 这种面向集合的语言有了面向过 程开发的能力。

在 SQL 中,游标是一种临时的数据库对象,可以指向存储在数据库表中的数据行指针。这里游标 充当了 指针的作用 ,我们可以通过操作游标来对数据行进行操作。

MySQL中游标可以在存储过程和函数中使用。

使用游标步骤

游标必须在声明处理程序之前被声明,并且变量和条件还必须在声明游标或处理程序之前被声明。 如果我们想要使用游标,一般需要经历四个步骤。不同的 DBMS 中,使用游标的语法可能略有不同。

第一步,声明游标

在MySQL中,使用DECLARE关键字来声明游标,其语法的基本形式如下:

DECLARE cursor_name CURSOR FOR select_statement;

 这个语法适用于 MySQL,SQL Server,DB2 和 MariaDB。如果是用 Oracle 或者 PostgreSQL,需要写成:

DECLARE cursor_name CURSOR IS select_statement;

 要使用 SELECT 语句来获取数据结果集,而此时还没有开始遍历数据,这里 select_statement 代表的是 SELECT 语句,返回一个用于创建游标的结果集。

比如:

DECLARE cur_emp CURSOR FOR

SELECT employee_id,salary FROM employees;

DECLARE cursor_fruit CURSOR FOR

SELECT f_name, f_price FROM fruits ;

第二步,打开游标

打开游标的语法如下:

OPEN cursor_name

当我们定义好游标之后,如果想要使用游标,必须先打开游标。打开游标的时候 SELECT 语句的查询结果集就会送到游标工作区,为后面游标的 逐条读取 结果集中的记录做准备。 

OPEN cur_emp ;

 第三步,使用游标(从游标中取得数据)

语法如下:

FETCH cursor_name INTO var_name [, var_name] ...

 这句的作用是使用 cursor_name 这个游标来读取当前行,并且将数据保存到 var_name 这个变量中,游 标指针指到下一行。如果游标读取的数据行有多个列名,则在 INTO 关键字后面赋值给多个变量名即可。

注意:var_name必须在声明游标之前就定义好。

FETCH cur_emp INTO emp_id, emp_sal ;

 注意:游标的查询结果集中的字段数,必须跟 INTO 后面的变量数一致,否则,在存储过程执行的时 候,MySQL 会提示错误。

第四步,关闭游标

CLOSE cursor_name

有 OPEN 就会有 CLOSE,也就是打开和关闭游标。当我们使用完游标后需要关闭掉该游标。因为游标会 占用系统资源 ,如果不及时关闭,游标会一直保持到存储过程结束,影响系统运行的效率。而关闭游标 的操作,会释放游标占用的系统资源。

关闭游标之后,我们就不能再检索查询结果中的数据行,如果需要检索只能再次打开游标

CLOSE cur_emp;

 小结

游标是 MySQL 的一个重要的功能,为 逐条读取 结果集中的数据,提供了完美的解决方案。跟在应用层 面实现相同的功能相比,游标可以在存储程序中使用,效率高,程序也更加简洁。 但同时也会带来一些性能问题,比如在使用游标的过程中,会对数据行进行 加锁 ,这样在业务并发量大 的时候,不仅会影响业务之间的效率,还会 消耗系统资源 ,造成内存不足,这是因为游标是在内存中进 行的处理。 建议:养成用完之后就关闭的习惯,这样才能提高系统的整体效率。 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值