mysql存儲(chǔ)過(guò)程 游標(biāo) 循環(huán)使用介紹
更新時(shí)間:2012年11月24日 19:37:34 作者:
今天分享下自己對(duì)于Mysql存儲(chǔ)過(guò)程的認(rèn)識(shí)與了解,這里主要說(shuō)說(shuō)大家常用的游標(biāo)加循環(huán)的嵌套使用
Mysql的存儲(chǔ)過(guò)程是從版本5才開(kāi)始支持的,所以目前一般使用的都可以用到存儲(chǔ)過(guò)程。今天分享下自己對(duì)于Mysql存儲(chǔ)過(guò)程的認(rèn)識(shí)與了解。
一些簡(jiǎn)單的調(diào)用以及語(yǔ)法規(guī)則這里就不在贅述,網(wǎng)上有許多例子。這里主要說(shuō)說(shuō)大家常用的游標(biāo)加循環(huán)的嵌套使用。
首先先介紹循環(huán)的分類:
(1)WHILE ... END WHILE
(2)LOOP ... END LOOP
(3)REPEAT ... END REPEAT
(4)GOTO
這里有三種標(biāo)準(zhǔn)的循環(huán)方式:WHILE循環(huán),LOOP循環(huán)以及REPEAT循環(huán)。還有一種非標(biāo)準(zhǔn)的循環(huán)方式:GOTO(不做介紹)。
(1)WHILE ... END WHILE
CREATE PROCEDURE p14()
BEGIN
DECLARE v INT;
SET v = 0;
WHILE v < 5 DO
INSERT INTO t VALUES (v);
SET v = v + 1;
END WHILE;
END;
這是WHILE循環(huán)的方式。它跟IF語(yǔ)句相似,使用"SET v = 0;"語(yǔ)句使為了防止一個(gè)常見(jiàn)的錯(cuò)誤,如果沒(méi)有初始化,默認(rèn)變量值為NULL,而NULL和任何值操作結(jié)果都為NULL。
(2)REPEAT ... END REPEAT
CREATE PROCEDURE p15 ()
BEGIN
DECLARE v INT;
SET v = 0;
REPEAT
INSERT INTO t VALUES (v);
SET v = v + 1;
UNTIL v >= 5
END REPEAT;
END;
這是REPEAT循環(huán)的例子,功能和前面WHILE循環(huán)一樣。區(qū)別在于它在執(zhí)行后檢查結(jié)果,而WHILE則是執(zhí)行前檢查。類似于do while語(yǔ)句。注意到UNTIL語(yǔ)句后面沒(méi)有分號(hào),在這里可以不寫(xiě)分號(hào),當(dāng)然你加上額外的分號(hào)更好。
(3)LOOP ... END LOOP
CREATE PROCEDURE p16 ()
BEGIN
DECLARE v INT;
SET v = 0;
loop_label: LOOP
INSERT INTO t VALUES (v);
SET v = v + 1;
IF v >= 5 THEN
LEAVE loop_label;
END IF;
END LOOP;
END;
以上是LOOP循環(huán)的例子。LOOP循環(huán)不需要初始條件,這點(diǎn)和WHILE循環(huán)相似,同時(shí)它又和REPEAT循環(huán)一樣也不需要結(jié)束條件。
ITERATE 迭代
如果目標(biāo)是ITERATE(迭代)語(yǔ)句的話,就必須用到LEAVE語(yǔ)句
CREATE PROCEDURE p20 ()
BEGIN
DECLARE v INT;
SET v = 0;
loop_label: LOOP
IF v = 3 THEN
SET v = v + 1;
ITERATE loop_label;
END IF;
INSERT INTO t VALUES (v);
SET v = v + 1;
IF v >= 5 THEN
LEAVE loop_label;
END IF;
END LOOP;
END;
ITERATE(迭代)語(yǔ)句和LEAVE語(yǔ)句一樣也是在循環(huán)內(nèi)部的循環(huán)引用, 它有點(diǎn)像C語(yǔ)言中 的“Continue”,同樣它可以出現(xiàn)在復(fù)合語(yǔ)句中,引用復(fù)合語(yǔ)句標(biāo)號(hào),ITERATE(迭代)意思 是重新開(kāi)始復(fù)合語(yǔ)句。
以上是對(duì)于循環(huán)的幾種情況的介紹。接著就是介紹一個(gè)帶游標(biāo)的例子來(lái)詳細(xì)解釋。
begin
declare p_feeCode varchar(20);
declare p_feeName varchar(20);
declare p_billMoney float(12);
declare p_schemeMoney float(12);
declare allMoney float(10);
declare allUsedMoney float(10);
declare p_year varchar(50);
declare p_totalCompeleteRate float(12);
declare done int(10);
declare flag int(2);
declare feeCodeCursor cursor for select feeCode from fee;//申明一個(gè)游標(biāo)變量
declare continue handler for not found set done=1;//申明循環(huán)結(jié)束的標(biāo)志位
set done=0;
select date_format(now(),'%Y') into p_year;
open feeCodeCursor;//打開(kāi)游標(biāo)
loop_label:LOOP
fetch feeCodeCursor into p_feeCode;//將游標(biāo)插入申明的變量
if done = 1 then
leave loop_label;
else
set flag = 0;
end if;
set p_schemeMoney=0;
set p_billMoney = 0;
select feeName into p_feeName from fee where feeCode=p_feeCode;
select sum(billMoney) into p_billMoney from bill_data where feeCode=p_feeCode and billDate like Concat(p_year, '%');
select schemeMoney into p_schemeMoney from total_scheme where feeCode=p_feeCode and schemeDate like Concat(p_year, '%') limit 1;
if flag = 0 then
set done = 0;
end if;
if p_schemeMoney=0 then
set p_totalCompeleteRate=-1.0;
else
set p_totalCompeleteRate=(1.0*p_billMoney)/p_schemeMoney;
end if;
insert into total_summary values(p_feeCode,p_feeName,p_year,p_billMoney,p_totalCompeleteRate);
commit;
end LOOP;
close feeCodeCursor;//循環(huán)結(jié)束后需要關(guān)閉游標(biāo)
end
以上只是一個(gè)簡(jiǎn)單的例子來(lái)說(shuō)明如何使用,大家不需要關(guān)注具體業(yè)務(wù)邏輯,只需要關(guān)注的是其中標(biāo)志位值的修改情況,已經(jīng)循環(huán)何時(shí)離開(kāi)。以及游標(biāo)如何聲明,如何使用,至于里面具體的操作和普通的sql語(yǔ)句沒(méi)有太大區(qū)別。此處是用一層循環(huán),至于復(fù)雜業(yè)務(wù)需要需要兩層三層,可以繼續(xù)用同樣的方法繼續(xù)嵌套。以下給出雙層嵌套循環(huán)的,同樣大家只需要關(guān)注嵌套結(jié)構(gòu)即可。
begin
declare p_projectID varchar(20);
declare p_projectName varchar(20);
declare p_feeCode varchar(20);
declare p_feeName varchar(20);
declare p_projectSchemeMoney float(10);
declare p_projectMoney float(10);
declare p_billMoney float(10);
declare p_year varchar(50);
declare p_projectFeeCompeleteRate float(10);
declare done1 int(10);
declare done2 int(10);
declare flag int(2);
declare feeCodeCursor cursor for select feeCode from fee;
declare continue handler for not found set done1=1;
set done1=0;
select date_format(now(),'%Y') into p_year;
delete from project_fee_summary;
open feeCodeCursor;
repeat //第一層嵌套開(kāi)始
fetch feeCodeCursor into p_feeCode;
select feeName into p_feeName from fee where feeCode=p_feeCode;
if not done1 then
begin
declare projectIDCursor cursor for select projectID from project;
declare continue handler for not found set done2 = 1;
set done2=0;
open projectIDCursor;
loop_label:LOOP//第二層嵌套開(kāi)始
fetch projectIDCursor into p_projectID;
select projectName into p_projectName from project where projectID=p_projectID;
if done2 = 1 then
leave loop_label;
else
set flag = 0;
end if;
if not done2 then
set p_projectSchemeMoney=0;
select sum(billMoney) into p_billMoney from bill_data where feeCode=p_feeCode and projectID=p_projectID and billDate like Concat(p_year, '%');
select projectSchemeMoney into p_projectSchemeMoney from project_scheme where feeCode=p_feeCode and projectID=p_projectID;
if flag = 0 then
set done2 = 0;
end if;
if p_projectSchemeMoney=0 then
set p_projectFeeCompeleteRate=-1;
else
set p_projectFeeCompeleteRate=(1.0*p_billMoney)/p_projectSchemeMoney;
end if;
insert into project_fee_summary values(p_feeCode,p_projectID,p_projectName,p_feeName,p_year,p_billMoney,p_projectFeeCompeleteRate,p_projectFeeCompeleteRate);
end if;
end LOOP;
select sum(billMoney) into p_projectMoney from bill_data where feeCode=p_feeCode and billDate like Concat(p_year, '%');
set p_projectFeeCompeleteRate=(1.0*p_projectMoney)/p_projectSchemeMoney;
insert into project_fee_summary values(p_feeCode,"total","total",p_feeName,p_year,p_projectMoney,p_projectFeeCompeleteRate,p_projectFeeCompeleteRate);
close projectIDCursor;
end;
end if;
until done1
end repeat;
close feeCodeCursor;
end
一些簡(jiǎn)單的調(diào)用以及語(yǔ)法規(guī)則這里就不在贅述,網(wǎng)上有許多例子。這里主要說(shuō)說(shuō)大家常用的游標(biāo)加循環(huán)的嵌套使用。
首先先介紹循環(huán)的分類:
(1)WHILE ... END WHILE
(2)LOOP ... END LOOP
(3)REPEAT ... END REPEAT
(4)GOTO
這里有三種標(biāo)準(zhǔn)的循環(huán)方式:WHILE循環(huán),LOOP循環(huán)以及REPEAT循環(huán)。還有一種非標(biāo)準(zhǔn)的循環(huán)方式:GOTO(不做介紹)。
(1)WHILE ... END WHILE
復(fù)制代碼 代碼如下:
CREATE PROCEDURE p14()
BEGIN
DECLARE v INT;
SET v = 0;
WHILE v < 5 DO
INSERT INTO t VALUES (v);
SET v = v + 1;
END WHILE;
END;
這是WHILE循環(huán)的方式。它跟IF語(yǔ)句相似,使用"SET v = 0;"語(yǔ)句使為了防止一個(gè)常見(jiàn)的錯(cuò)誤,如果沒(méi)有初始化,默認(rèn)變量值為NULL,而NULL和任何值操作結(jié)果都為NULL。
(2)REPEAT ... END REPEAT
復(fù)制代碼 代碼如下:
CREATE PROCEDURE p15 ()
BEGIN
DECLARE v INT;
SET v = 0;
REPEAT
INSERT INTO t VALUES (v);
SET v = v + 1;
UNTIL v >= 5
END REPEAT;
END;
這是REPEAT循環(huán)的例子,功能和前面WHILE循環(huán)一樣。區(qū)別在于它在執(zhí)行后檢查結(jié)果,而WHILE則是執(zhí)行前檢查。類似于do while語(yǔ)句。注意到UNTIL語(yǔ)句后面沒(méi)有分號(hào),在這里可以不寫(xiě)分號(hào),當(dāng)然你加上額外的分號(hào)更好。
(3)LOOP ... END LOOP
復(fù)制代碼 代碼如下:
CREATE PROCEDURE p16 ()
BEGIN
DECLARE v INT;
SET v = 0;
loop_label: LOOP
INSERT INTO t VALUES (v);
SET v = v + 1;
IF v >= 5 THEN
LEAVE loop_label;
END IF;
END LOOP;
END;
以上是LOOP循環(huán)的例子。LOOP循環(huán)不需要初始條件,這點(diǎn)和WHILE循環(huán)相似,同時(shí)它又和REPEAT循環(huán)一樣也不需要結(jié)束條件。
ITERATE 迭代
如果目標(biāo)是ITERATE(迭代)語(yǔ)句的話,就必須用到LEAVE語(yǔ)句
復(fù)制代碼 代碼如下:
CREATE PROCEDURE p20 ()
BEGIN
DECLARE v INT;
SET v = 0;
loop_label: LOOP
IF v = 3 THEN
SET v = v + 1;
ITERATE loop_label;
END IF;
INSERT INTO t VALUES (v);
SET v = v + 1;
IF v >= 5 THEN
LEAVE loop_label;
END IF;
END LOOP;
END;
ITERATE(迭代)語(yǔ)句和LEAVE語(yǔ)句一樣也是在循環(huán)內(nèi)部的循環(huán)引用, 它有點(diǎn)像C語(yǔ)言中 的“Continue”,同樣它可以出現(xiàn)在復(fù)合語(yǔ)句中,引用復(fù)合語(yǔ)句標(biāo)號(hào),ITERATE(迭代)意思 是重新開(kāi)始復(fù)合語(yǔ)句。
以上是對(duì)于循環(huán)的幾種情況的介紹。接著就是介紹一個(gè)帶游標(biāo)的例子來(lái)詳細(xì)解釋。
復(fù)制代碼 代碼如下:
begin
declare p_feeCode varchar(20);
declare p_feeName varchar(20);
declare p_billMoney float(12);
declare p_schemeMoney float(12);
declare allMoney float(10);
declare allUsedMoney float(10);
declare p_year varchar(50);
declare p_totalCompeleteRate float(12);
declare done int(10);
declare flag int(2);
declare feeCodeCursor cursor for select feeCode from fee;//申明一個(gè)游標(biāo)變量
declare continue handler for not found set done=1;//申明循環(huán)結(jié)束的標(biāo)志位
set done=0;
select date_format(now(),'%Y') into p_year;
open feeCodeCursor;//打開(kāi)游標(biāo)
loop_label:LOOP
fetch feeCodeCursor into p_feeCode;//將游標(biāo)插入申明的變量
if done = 1 then
leave loop_label;
else
set flag = 0;
end if;
set p_schemeMoney=0;
set p_billMoney = 0;
select feeName into p_feeName from fee where feeCode=p_feeCode;
select sum(billMoney) into p_billMoney from bill_data where feeCode=p_feeCode and billDate like Concat(p_year, '%');
select schemeMoney into p_schemeMoney from total_scheme where feeCode=p_feeCode and schemeDate like Concat(p_year, '%') limit 1;
if flag = 0 then
set done = 0;
end if;
if p_schemeMoney=0 then
set p_totalCompeleteRate=-1.0;
else
set p_totalCompeleteRate=(1.0*p_billMoney)/p_schemeMoney;
end if;
insert into total_summary values(p_feeCode,p_feeName,p_year,p_billMoney,p_totalCompeleteRate);
commit;
end LOOP;
close feeCodeCursor;//循環(huán)結(jié)束后需要關(guān)閉游標(biāo)
end
以上只是一個(gè)簡(jiǎn)單的例子來(lái)說(shuō)明如何使用,大家不需要關(guān)注具體業(yè)務(wù)邏輯,只需要關(guān)注的是其中標(biāo)志位值的修改情況,已經(jīng)循環(huán)何時(shí)離開(kāi)。以及游標(biāo)如何聲明,如何使用,至于里面具體的操作和普通的sql語(yǔ)句沒(méi)有太大區(qū)別。此處是用一層循環(huán),至于復(fù)雜業(yè)務(wù)需要需要兩層三層,可以繼續(xù)用同樣的方法繼續(xù)嵌套。以下給出雙層嵌套循環(huán)的,同樣大家只需要關(guān)注嵌套結(jié)構(gòu)即可。
復(fù)制代碼 代碼如下:
begin
declare p_projectID varchar(20);
declare p_projectName varchar(20);
declare p_feeCode varchar(20);
declare p_feeName varchar(20);
declare p_projectSchemeMoney float(10);
declare p_projectMoney float(10);
declare p_billMoney float(10);
declare p_year varchar(50);
declare p_projectFeeCompeleteRate float(10);
declare done1 int(10);
declare done2 int(10);
declare flag int(2);
declare feeCodeCursor cursor for select feeCode from fee;
declare continue handler for not found set done1=1;
set done1=0;
select date_format(now(),'%Y') into p_year;
delete from project_fee_summary;
open feeCodeCursor;
repeat //第一層嵌套開(kāi)始
fetch feeCodeCursor into p_feeCode;
select feeName into p_feeName from fee where feeCode=p_feeCode;
if not done1 then
begin
declare projectIDCursor cursor for select projectID from project;
declare continue handler for not found set done2 = 1;
set done2=0;
open projectIDCursor;
loop_label:LOOP//第二層嵌套開(kāi)始
fetch projectIDCursor into p_projectID;
select projectName into p_projectName from project where projectID=p_projectID;
if done2 = 1 then
leave loop_label;
else
set flag = 0;
end if;
if not done2 then
set p_projectSchemeMoney=0;
select sum(billMoney) into p_billMoney from bill_data where feeCode=p_feeCode and projectID=p_projectID and billDate like Concat(p_year, '%');
select projectSchemeMoney into p_projectSchemeMoney from project_scheme where feeCode=p_feeCode and projectID=p_projectID;
if flag = 0 then
set done2 = 0;
end if;
if p_projectSchemeMoney=0 then
set p_projectFeeCompeleteRate=-1;
else
set p_projectFeeCompeleteRate=(1.0*p_billMoney)/p_projectSchemeMoney;
end if;
insert into project_fee_summary values(p_feeCode,p_projectID,p_projectName,p_feeName,p_year,p_billMoney,p_projectFeeCompeleteRate,p_projectFeeCompeleteRate);
end if;
end LOOP;
select sum(billMoney) into p_projectMoney from bill_data where feeCode=p_feeCode and billDate like Concat(p_year, '%');
set p_projectFeeCompeleteRate=(1.0*p_projectMoney)/p_projectSchemeMoney;
insert into project_fee_summary values(p_feeCode,"total","total",p_feeName,p_year,p_projectMoney,p_projectFeeCompeleteRate,p_projectFeeCompeleteRate);
close projectIDCursor;
end;
end if;
until done1
end repeat;
close feeCodeCursor;
end
相關(guān)文章
Mysql表數(shù)據(jù)比較大情況下修改添加字段的方法實(shí)例
最近遇到的一個(gè)問(wèn)題,需要在一張1800萬(wàn)數(shù)據(jù)量的表中添加加一個(gè)字段,所以這篇文章主要給大家介紹了關(guān)于Mysql表數(shù)據(jù)比較大情況下修改添加字段的方法,需要的朋友可以參考下2022-06-06MySQL內(nèi)連接和外連接及七種SQL?JOINS的實(shí)現(xiàn)
這篇文章主要介紹了Mysql內(nèi)連接和外連接的區(qū)別以及七種SQL?Joins的實(shí)現(xiàn),相信看完這篇文章你對(duì)SQL內(nèi)外連接的多表查詢就足夠理解了,需要的朋友可以參考下2023-03-03MySQL學(xué)習(xí)之基礎(chǔ)命令實(shí)操總結(jié)
MySQL 是最流行的關(guān)系型數(shù)據(jù)庫(kù)管理系統(tǒng),在WEB應(yīng)用方面MySQL是最好的。本文將為大家詳細(xì)介紹一些MySQL的基礎(chǔ)命令,需要的可以參考一下2022-03-03解決Mysql主從錯(cuò)誤:could not find first log&nbs
這篇文章主要介紹了解決Mysql主從錯(cuò)誤:could not find first log file name in binary問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-12-12設(shè)置MySQL中的數(shù)據(jù)類型來(lái)優(yōu)化運(yùn)行速度的實(shí)例
這篇文章主要介紹了設(shè)置MySQL中索引的數(shù)據(jù)類型來(lái)優(yōu)化運(yùn)行速度的實(shí)例,主要是適當(dāng)使用短字節(jié)的數(shù)據(jù)類型來(lái)處理短索引,需要的朋友可以參考下2015-05-05MySQL數(shù)據(jù)庫(kù)引擎介紹、區(qū)別、創(chuàng)建和性能測(cè)試的深入分析
本篇文章是對(duì)MySQL數(shù)據(jù)庫(kù)引擎介紹、區(qū)別、創(chuàng)建和性能測(cè)試進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-06-06