一、游标的概念
? 游标是一个存储在MySQL服务器上的数据库查询,它不是一条select语句,而是被该语句检索出来的结果集。有了游标可以方便的对该结果集进行逐行处理。
二、游标的使用
1. 创建游标
-- declare语句定义了名为ordernumbers的游标。存储过程处理完成后,游标就会消失(因为它局限于存储过程)。
create procedure processorders()
begin
declare ordernumbers cursor
for
select order_num from orders;
end;
【注】不像多数DBMS,MySQL游标只能用于存储过程(和函数)。
2. 打开和关闭游标
-- 打开游标。在处理open语句时执行上面的select查询语句,存储检索出的数据以供浏览和滚动。
open ordernumbers;
-- 游标处理完成后,利用close语句释放游标使用的所有内部内存和资源。
close ordernumbers;
【注】当一个游标关闭后,如果需要再次使用它,可以使用open语句再次打开它。
? 如果你不明确关闭游标,MySQL将会在到达end语句时自动关闭它。
下面是前面例子的修改版本:
create procedure processorders()
begin
declare ordernumbers cursor
for
select order_num from orders;
open ordernumbers;
close ordernumbers;
end;
-- 这个存储过程声明、打开和关闭一个游标。但对检索出的数据什么也没做。
3. 使用游标数据
第一个例子从游标中检索第一行:
create procedure processorders()
begin
declare o int;
declare ordernumbers cursor
for
select order_num from orders;
open ordernumbers;
-- 利用fetch检索出第一行的order_num存储到一个名为o的局部变量中。
fetch ordernumbers into o;
close ordernumbers;
end;
第二个例子检索游标中的所有行,从第一行到最后一行:
create procedure processorders()
begin
declare done boolean default 0;
declare o int;
declare ordernumbers cursor
for
select order_num from orders;
/*
这条语句定义了一个continue handler,它是在条件出现时被执行的代码。这里,它指出当
sqlstate ‘02000‘出现时,set done = 1。sqlstate ‘02000‘是一个未找到条件,当
repeate由于没有更多的行供循环而不能继续时,出现这个条件。
*/
declare continue handler for sqlstate ‘02000‘ set done = 1;
open ordernumbers;
-- 当done为真(非零)时结束循环。
repeat
fetch ordernumbers into o;
until done end repeat;
close ordernumbers;
end;
第三个例子对从游标中取出的数据进行某种实际的处理
create procedure processorders()
begin
declare done boolean default 0;
declare o int;
declare t decimal(8,2);
declare ordernumbers cursor
for
select order_num from orders;
declare continue handler for sqlstate ‘02000‘ set done = 1;
-- 创建一个表用来存放结果
create table if not exists ordertotals
(order_num int, total decimal(8,2));
open ordernumbers;
repeat
fetch ordernumbers into o;
-- ordertotal为在上一章创建的一个用来计算带税合计的存储过程
call ordertotal(o, 1, t);
insert into ordertotals(order_num, total)
values(o, t);
until done end repeat;
close ordernumbers;
end;
-- 此存储过程不返回数据,但它能创建和填充另一个表。