1.游标说明

游标,有些地方也称为光标。它的作用是在一个结果集中逐条逐条地获取记录行并操作它们。

例如:

其中select是游标所操作的结果集,游标每次fetch一行中的name和age字段,并将每一行的这两个字段赋值给变量var1和var2。

有很多、很多、很多人,很多、很多、很多书都强烈建议:能不用游标尽量不要用游标。因为它违背了集合的理论,集合取数据是一把一把抓,游标取数据的时候一行一行取,每取一行操作一行,而且在每一行上都有额外的资源消耗。总之,游标效率低、资源消耗高。

其实很多领域都有这样的优化:把数据先集中起来,集中到了一定量再一次性处理,这样的处理方式效率要高得多。比如写日志到磁盘上,可以每产生一条日志就刷入磁盘,也可以先产生一堆日志缓存起来,之后一次性刷如磁盘。后者效率要高得多。

集合取数据的时候关注点在于想要什么数据,而不关注怎么去获取数据,游标的关注点则在于怎么获取这些数据:将游标指针作为遍历依据,遍历到哪行数据就返回这行数据然后停下来处理数据,再继续遍历数据。习惯于迭代的人比较喜欢游标,特别是习惯C语言的人,因为游标就是遍历数据行的行为。

在MySQL、MariaDB中实现的游标比较简单,它只有一种遍历方式:逐行向前遍历。MariaDB 10.3后,游标方面支持的更完整一点:支持游标参数。

光标的使用包括声明光标、打开光标、使用光标和关闭光标(MySQL/MariaDB中的游标无需释放)。光标必须声明在处理程序之前,并且在声明保存结果集的变量之后。另外,游标是一种复合语句结构(就像begin...end),只能用于stored procedure或stored function中。

2.使用游标

1.声明游标

DECLARE cursor_name CURSOR FOR select_statement;

其中select_statement是游标需要从中获取的结果集。

例如:

declare cur_city cursor for select id,'name',population from world,city;

在MariaDB 10.3中,支持游标参数,该参数可以传递到select_statement中:

DECLARE cursor_name CURSOR(param1 data_type,param2 data_type2...) FOR select_statement;

例如:

declare cur_stu cursor(min int,max int) for select id,name from Student where id between min and max;

注意,mariaDB 10.3之前的语法也能在10.3版本上执行,因为之前的语法是10.3版本中不带参数的特殊情况。

2.声明处理程序

一般来说,光标是用在逐条取结果集的情况下,所以在使用光标的时候基本都会放在循环结构中循环获取数据存储到变量中。但如何在取完数据后退出循环?

在游标无法获取到下一行数据的时候,将会返回一个1329错误码,这个错误码对应的SQL状态码为"02000",它们等价于NOT FOUND(这几个是等价的,只是MariaDB中分了3类描述问题的代码而已)。这时可以在声明游标后定义一个handler,用于处理NOT FOUND。

例如下面是适合游标NOT FOUND时的CONTINUE处理器,表示当找不到下一行数据时继续执行后面的程序:

DECLARE CONTINUE HANDLER FOR NOT FOUND statement;

对于处理游标的HANDLER,通常statement部分是SET语句,用于设置一些变量。例如:

declare continue handler for not found set var_name=value;

这时,当取不到下一条记录时即已经取完记录时,就设置变量var_name=value。之后就可以通过该变量的值作为退出循环的条件。

关于handler详细内容,见我翻译的MariaDB手册:https://mariadb.com/kb/zh-cn/declare-handler/

3.打开游标

当声明了一个游标后,必须要打开游标才能使用游标。

open cursor_name;

例如:

open cur_city;

对于mariadb 10.3,由于支持游标参数,因此语法为:

open cursor_name(value1,value2);

例如:

open cur_stu(4,10);

4.使用游标(fetch into)

通过fetch into命令将每次fetch到的结果存储到预先定义好的变量中。注意,这个变量必须是本地变量(局部变量),不能是用户自定义变量,且这个变量必须定义在游标声明语句之前。

fetch cursor_name into var_name;

例如:

fetch cur_city into city_id,city_name,city_popcnt;

在上面已经说过了,一般游标都会在循环结构中使用。以下是在repeat结构中使用游标;

repeat
fetch ... into ...
until var_name=value
end repeat;

5.关闭游标

close cursor_name;

例如:

close cur_city;

3.游标使用示例

以下是MariaDB 10.3版本之前(也适用于10.3)的游标使用示例:将表t1和表t2中每行中的某一列作比较,将较大值插入到表t3中。

create or replace table t1(i int);
create or replace table t2(i int);
create or replace table t3(i int); insert into t1 values(5),(10),(20);
insert into t2 values(15),(30),(10); delimiter $$ create or replace procedure proc1()
begin
declare done int default false; /* 用于判断退出循环 */
declare x,y int; /* 用于保存fetch结果 */
declare cur1 cursor for select i from t1; /* fetch t1的游标 */
declare cur2 cursor for select i from t2; /* fetch t2的游标 */
declare continue handler for not found set done=true; /* not found时,退出循环 */ open cur1;
open cur2; my_loop: LOOP
fetch cur1 into x;
fetch cur2 into y;
if done then
leave my_loop;
end if;
if x <= y then
insert into t3 values(y);
else
insert into t3 values(x);
end if;
end loop; close cur1;
close cur2;
end$$ delimiter ; call proc1;

查看表t3:

select * from t3;
+------+
| i |
+------+
| 15 |
| 30 |
| 20 |
+------+

下面是MariaDB 10.3上使用游标的一个示例:将表t1中i字段某一段数据插入到表t2中。

create or replace table t1(i int);
create or replace table t2(i int); insert into t1 values(5),(10),(20),(30),(40); delimiter $$ create or replace procedure proc1(min int,max int)
begin
declare done int default false;
declare x int;
declare cur1 cursor(cmin int,cmax int) for select i from t1 where t1.i between cmin and cmax;
declare continue handler for not found set done=true; open cur1(min,max); my_loop: LOOP
fetch cur1 into x;
if done then
leave my_loop;
end if;
insert into t2 values(x);
end loop; close cur1;
end$$ delimiter ; call proc1(10,40);

查看t2结果:

MariaDB [test]> select * from t2;
+------+
| i |
+------+
| 10 |
| 20 |
| 30 |
| 40 |
+------+

MySQL/MariaDB中游标的使用的更多相关文章

  1. MySQL/MariaDB/PerconaDB-提权条件竞争漏洞

    背景 2016年11月01日,国外安全研究员Dawid Golunski在 MySQl, MariaDB 和 PerconaDB 数据库中发现条件竞争漏洞,该漏洞允许本地用户使用低权限(CREATE/ ...

  2. 15 个有用的 MySQL/MariaDB 性能调整和优化技巧(转载的一篇好文)

    MySQL 是一个强大的开源关系数据库管理系统(简称 RDBMS).它发布于 1995 年(20年前).它采用结构化查询语言(SQL),这可能是数据库内容管理中最流行的选择.最新的 MySQL 版本是 ...

  3. 【原】基于 HAproxy 1.6.3 Keeplived 在 Centos 7 中实现mysql mariadb galera cluster 集群分发读写 —— 上篇

    前言 有一段时间没有写blogs,乘着周末开始整理下haproxy + keeplived 实现 mysql mariadb galera cluster 集群访问环境的搭建工作. 本文集中讲hapr ...

  4. 在CentOS 7 MySQL / MariaDB

    在CentOS7中,MariaDB  替代了MySQL;更多复杂的疑问可以在这里查看 MariaDB versus MySQL – Compatibility Install MySQL / Mari ...

  5. MySQL/MariaDB/Percona数据库升级脚本

    MySQL/MariaDB/Percona数据库升级脚本截取<OneinStack>中upgrade_db.sh, 一般情况下不建议升级数据库版本,该脚本专提供给各位版本控们.为防止大版本 ...

  6. 流程控制语句(MySQL/MariaDB )

    本文目录:1.BEGIN...END2.true和false3.if结构4.case结构5.loop.leave和iterate6.repeat循环7.while循环 MySQL/MariaDB中的符 ...

  7. MySQL/MariaDB触发器

    本文目录:1.创建触发器2.insert触发器3.delete触发器4.update触发器5.通过on duplicate key update分析触发器触发原理6.replace to算法验证7.查 ...

  8. MySQL/MariaDB的锁

    本文目录: 1.MariaDB/MySQL事务提交的方式 2.MariaDB/MySQL中的锁简介 2.1 不同存储引擎支持的锁级别 2.2 锁类型 2.3 锁兼容性 3.MyISAM的表级锁(loc ...

  9. mysql(mariadb)如何更改root密码

    mysql(或者mariadb,她是mysql的一个分支,完全开源,新版本的linux系统默认安装的是mariadb)如何更改root密码呢?我们主要介绍命令mysqladmin来实现. mysql( ...

随机推荐

  1. telnet命令使用详解

    telnet命令用于登录远程主机,对远程主机进行管理.telnet因为采用明文传送报文,安全性不好,很多Linux服务器都不开放telnet服务,而改用更安全的ssh方式了.但仍然有很多别的系统可能采 ...

  2. 关于LINUX各类系统资源整合

    在系统维护的过程中,随时可能有需要查看 CPU和内存的使用率,并根据相应信息分析系统状况的需求.本文介绍一下几种常见的Linux系统资源查看命令. 1.总体内存占用的查看 命令:free 图1 fre ...

  3. View滑动的常见方式

    今天头条的面试官问起了实现View滑动的方式,其实前几天刚刚看过,但还是没能答上来,这里再来总结一遍. 1.使用scrollTo/scrollBy 为了实现View滑动,Android专门提供了这两个 ...

  4. 小米google play service停止工作解决办法,不root,不刷第三方recovery(也适用于其他的手机)

    问题: 原因是手机安卓系统是6.0.系统应用里面没有包含谷歌框架等一系列谷歌的小东西. 参考: http://www.miui.com/thread-3548436-1-1.html http://w ...

  5. Spring依赖注入 — util命名空间配置

    要使用<util>标签,必须在XML中加入util名称空间(namespace): xmlns:util="http://http://www.springframework.o ...

  6. async generator promise异步方案实际运用

    es7 async方案 /******************async***********************/ var timeFn=function(time){ return new P ...

  7. 设计模式之观察者(OBSERVER)模式

    定义 定义对象间的一种一对多的依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都得到通知并被自动更新.  Observer模式描述了如何建立这种关系.这一模式中的关键对象是目标(subject ...

  8. Vue之九数据劫持实现MVVM的数据双向绑定

    vue是通过数据劫持的方式来做数据绑定的,其中最核心的方法便是通过Object.defineProperty()来实现对属性的劫持,达到监听数据变动的目的. 如果不熟悉defineProperty,猛 ...

  9. python基础学习二 数据结构之list及相关基本操作

    list是py内置的一种数据类型,list就是列表的意思,list就是一种有序的数据集合,可以随时增加和删除list的元素. 生活中,比如我们要列出全班同学的名字,就可以用list来表示 >&g ...

  10. Android类参考---SQLiteOpenHelper

    public 抽象类 SQLiteOpenHelper 继承关系 java.lang.Object |____android.database.sqlite.SQLiteOpenHelper 类概要 ...