MySQL游標(biāo)概念與用法詳解
本文實(shí)例講述了MySQL游標(biāo)概念與用法。分享給大家供大家參考,具體如下:
1、游標(biāo)的概念(Cursor)
一條sql,對(duì)應(yīng)N條資源,取出資源的接口,就是游標(biāo),沿著游標(biāo),可以一次取出1行。如果開(kāi)發(fā)過(guò)安卓的同學(xué)應(yīng)該知道有一個(gè)Api是Cursor,也是讀取SQLite數(shù)據(jù)庫(kù)用的,和這個(gè)有點(diǎn)類似。
2、使用游標(biāo)的步驟
(1)聲明
使用declare進(jìn)行聲明
declare 游標(biāo)名 cursor for select_statement
(2)打開(kāi)游標(biāo)
使用open進(jìn)行打開(kāi)
open 游標(biāo)名
(3)從游標(biāo)中取值
使用fetch進(jìn)行取值
fetch 游標(biāo)名 into var1,var2[,...] --將取到的一行賦值給多個(gè)變量
(4)關(guān)閉游標(biāo)
使用close關(guān)閉游標(biāo)
close 游標(biāo)名
3、創(chuàng)建一個(gè)簡(jiǎn)單的游標(biāo)
需求:從商品表中讀取第一行數(shù)據(jù)
商品表(goods)數(shù)據(jù):

注意:我這里已經(jīng)將MySQL的結(jié)束標(biāo)識(shí)符改為 $,如果要知道怎么設(shè)置為$,請(qǐng)參考前面一篇文章:MySQL觸發(fā)器。
定義:
create procedure p12() begin /*定義三個(gè)變量用于存放商品id,商品名稱,商品庫(kù)存量*/ declare row_gid int ; declare row_name varchar(20); declare row_num int; declare getgoods cursor for select gid,name,num from goods; --定義游標(biāo) open getgoods; --打開(kāi)游標(biāo) fetch getgoods into row_gid,row_name,row_num;--從游標(biāo)中取值 select row_name,row_num; --顯示操作 close getgoods; --關(guān)閉游標(biāo) end$
輸出結(jié)果:

4、多次取值操作
create procedure p13() begin declare row_gid int ; declare row_name varchar(20); declare row_num int; declare getgoods cursor for select gid,name,num from goods; open getgoods; fetch getgoods into row_gid,row_name,row_num; select row_name,row_num; fetch getgoods into row_gid,row_name,row_num; select row_name,row_num; fetch getgoods into row_gid,row_name,row_num; select row_name,row_num; fetch getgoods into row_gid,row_name,row_num; select row_name,row_num; close getgoods; end$
輸出:

注意:當(dāng)游標(biāo)讀到末尾,如果繼續(xù)進(jìn)行取值操作會(huì)發(fā)生報(bào)錯(cuò)
5、游標(biāo)循環(huán)表中的所有數(shù)據(jù)
(1)使用計(jì)數(shù)器來(lái)循環(huán)
create procedure p14()
begin
declare cnt int default 0;
declare i int default 0;
declare row_gid int ;
declare row_name varchar(20);
declare row_num int;
declare getgoods cursor for select gid,name,num from goods;
select count(*) into cnt from goods;
open getgoods;
repeat
fetch getgoods into row_gid,row_name,row_num;
select row_name,row_num;
set i:= i+1;
until i >= cnt end repeat;
close getgoods;
end$
輸出結(jié)果:

(2)使用越界標(biāo)志來(lái)控制循環(huán)
在mysql cursor中,可以聲明declare continue handler來(lái)操作1個(gè)越界標(biāo)志
語(yǔ)法:
declare continue handler for NOT FOUND statement;
使用:
create procedure p15()
begin
declare row_gid int ;
declare row_name varchar(20);
declare row_num int;
declare have int default 1;
declare getgoods cursor for select gid,name,num from goods;
declare continue handler for NOT FOUND set have:= 0;
open getgoods;
repeat
fetch getgoods into row_gid,row_name,row_num;
select row_name,row_num;
until have = 0 end repeat;
close getgoods;
end$
輸出結(jié)果:

注意:這里發(fā)生了錯(cuò)誤,這里輸出了4行數(shù)據(jù),而表中只有3行數(shù)據(jù),而且還爆出了警告,后面會(huì)說(shuō)怎么結(jié)果這個(gè)問(wèn)題。
程序執(zhí)行邏輯:
循環(huán)游標(biāo)->fetch第三條數(shù)據(jù)->顯示->fetch第四條數(shù)據(jù)->沒(méi)有數(shù)據(jù)->設(shè)置have=0操作->執(zhí)行continue Handler->程序不退出,執(zhí)行顯示操作->還是顯示第三條數(shù)據(jù)
6、continue和exit的區(qū)別
continue:若沒(méi)有數(shù)據(jù)返回,程序繼續(xù),并將變量IS_FOUND設(shè)為0,這種情況是出現(xiàn)在select XX into XXX from tablename的時(shí)候發(fā)生的。
exit:若沒(méi)有數(shù)據(jù)返回,退出程序,并將變量IS_FOUND設(shè)為0,這種情況是出現(xiàn)在select XX into XXX from tablename的時(shí)候發(fā)生的。
使用exit來(lái)替換continue:
使用exit就不會(huì)出現(xiàn)上面的那種情況了,程序執(zhí)行邏輯:
循環(huán)游標(biāo)->fetch到第三條數(shù)據(jù)->顯示->第四次fetch操作->沒(méi)有數(shù)據(jù)->設(shè)置 have=0操作->程序直接退出exit
所以就沒(méi)有顯示出第四條數(shù)據(jù)。
create procedure p16()
begin
declare row_gid int ;
declare row_name varchar(20);
declare row_num int;
declare have int default 1;
declare getgoods cursor for select gid,name,num from goods;
declare exit handler for NOT FOUND set have:= 0;
open getgoods;
repeat
fetch getgoods into row_gid,row_name,row_num;
select row_name,row_num;
until have = 0 end repeat;
close getgoods;
end$
輸出結(jié)果:

7、正確的游標(biāo)循環(huán)
在一些特殊的情況中,我們可以讀到的數(shù)據(jù)為空,或者壓根sql語(yǔ)句就有錯(cuò)誤,我們不能避免出現(xiàn)這種情況,所以我們要正確的使用游標(biāo)循環(huán)操作。
首先應(yīng)該創(chuàng)建游標(biāo),然后打開(kāi)游標(biāo)后,應(yīng)先手動(dòng)進(jìn)行fetch操作獲取到一行數(shù)據(jù),然后再通過(guò)循環(huán),在循環(huán)里先做處理內(nèi)容,后進(jìn)行fetch操作。這樣如果在手動(dòng)獲取數(shù)據(jù)的期間就沒(méi)有獲得到數(shù)據(jù)的話,就會(huì)執(zhí)行have = 0,如果是repeat循環(huán),然后進(jìn)入repeat循環(huán),先輸出null數(shù)據(jù),最后又進(jìn)行獲取,這樣運(yùn)行到until時(shí)就會(huì)退出循環(huán);如果是while循環(huán),壓根就不進(jìn)去while循環(huán)里,就不會(huì)有任何1行輸出。
(1)repeat循環(huán):
create procedure p17()
begin
declare row_gid int;
declare row_name varchar(20);
declare row_num int;
declare have int default 1;
declare getgoods cursor for select gid,name,num from goods where 0;
declare continue handler for NOT FOUND set have:= 0;
open getgoods;
fetch getgoods into row_gid,row_name,row_num;
repeat
select row_name,row_num;
fetch getgoods into row_gid,row_name,row_num;
until have = 0 end repeat;
close getgoods;
end$
輸出結(jié)果:

(2)while循環(huán):
create procedure p18()
begin
declare row_gid int;
declare row_name varchar(20);
declare row_num int;
declare have int default 1;
declare getgoods cursor for select gid,name,num from goods where 0;
declare continue handler for NOT FOUND set have:= 0;
open getgoods;
fetch getgoods into row_gid,row_name,row_num;
while have = 1 do
select row_name,row_num;
fetch getgoods into row_gid,row_name,row_num;
end while;
close getgoods;
end$
輸出結(jié)果:

更多關(guān)于MySQL相關(guān)內(nèi)容感興趣的讀者可查看本站專題:《MySQL查詢技巧大全》、《MySQL事務(wù)操作技巧匯總》、《MySQL存儲(chǔ)過(guò)程技巧大全》、《MySQL數(shù)據(jù)庫(kù)鎖相關(guān)技巧匯總》及《MySQL常用函數(shù)大匯總》
希望本文所述對(duì)大家MySQL數(shù)據(jù)庫(kù)計(jì)有所幫助。
相關(guān)文章
手把手教你使用Navicat生成MySQL測(cè)試數(shù)據(jù)
Navicat是一套快速、可靠并價(jià)格相當(dāng)便宜的數(shù)據(jù)庫(kù)管理工具,專為簡(jiǎn)化數(shù)據(jù)庫(kù)的管理及降低系統(tǒng)管理成本而設(shè),下面這篇文章主要給大家介紹了關(guān)于使用Navicat生成MySQL測(cè)試數(shù)據(jù)的相關(guān)資料,需要的朋友可以參考下2023-05-05
Mysql數(shù)據(jù)庫(kù)group?by原理詳解
這篇文章主要為大家介紹了Mysql數(shù)據(jù)庫(kù)group?by的原理詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-07-07
MySQL實(shí)戰(zhàn)文章(非常全的基礎(chǔ)入門(mén)類教程)
半個(gè)月時(shí)間把MySQL重新鞏固了一遍,梳理了一篇幾萬(wàn)字超硬核文章,想學(xué)習(xí)mysql的朋友可以看看2023-05-05
VS2022連接數(shù)據(jù)庫(kù)MySQL并進(jìn)行基本的表的操作指南
鑒于MySQL數(shù)據(jù)庫(kù)的流行與強(qiáng)大,決定多學(xué)習(xí)使用,下面這篇文章主要給大家介紹了關(guān)于VS2022連接數(shù)據(jù)庫(kù)MySQL并進(jìn)行基本的表的操作指南,文中通過(guò)圖文介紹的非常詳細(xì),需要的朋友可以參考下2023-05-05
生產(chǎn)環(huán)境MySQL索引時(shí)效的排查過(guò)程
這篇文章主要介紹了生產(chǎn)環(huán)境MySQL索引時(shí)效的排查過(guò)程,文章根據(jù)SQL查詢耗時(shí)特別長(zhǎng),看了執(zhí)行計(jì)劃發(fā)現(xiàn)沒(méi)有走索引的問(wèn)題展開(kāi)詳細(xì)介紹,需要的朋友可以參考一下2022-04-04

