下面給大家介紹在什么情況下用隱式游標(biāo),什么情況下用顯示游標(biāo):
1.查詢返回單行記錄時→隱式游標(biāo);
2.查詢返回多行記錄并逐行進行處理時→顯式游標(biāo)
--顯示游標(biāo)屬性
declare
CURSOR cur_emp IS SELECT * FROM emp;
row_emp cur_emp%ROWTYPE;
BEGIN
OPEN cur_emp;
FETCH cur_emp INTO row_emp;
WHILE cur_emp%FOUND
LOOP
dbms_output.put_line(row_emp.empno||'----'||row_emp.ename);
FETCH cur_emp INTO row_emp;
END LOOP;
close cur_emp;
END;
--使用顯式游標(biāo)修改數(shù)據(jù)(給所有的部門經(jīng)理加薪1000)
DECLARE
CURSOR emp_cur IS
SELECT empno,ename,sal FROM emp WHERE job='MANAGER' FOR UPDATE;
emp_row emp_cur%ROWTYPE;
BEGIN
OPEN emp_cur;
LOOP
FETCH emp_cur INTO emp_row;
IF emp_cur%NOTFOUND THEN
EXIT;
ELSE
UPDATE emp SET sal=sal+1000 WHERE CURRENT OF emp_cur;
END IF;
END LOOP;
COMMIT;
CLOSE emp_cur;
END;
·注意:
1、如果游標(biāo)打開之前或關(guān)閉之后,使用游標(biāo)屬性,Oracle會拋出一個INVALID_CURSOR錯誤(ORA-01001);
2、如果在第一次fetch后結(jié)果集是空的,%found=false,%NotFound=true,%ROWCOUNT=0;
3、如果使用了BULK COLLECT,那么%ROWCOUNT的值可能不是0或1,實際上他返回的是提取到相關(guān)集合的行數(shù)。
--游標(biāo)for循環(huán)(給所有的部門經(jīng)理減薪1000)
DECLARE
CURSOR emp_cur IS
SELECT empno,ename,sal FROM emp WHERE job='MANAGER' FOR UPDATE;
BEGIN
FOR emp_row IN emp_cur
LOOP
UPDATE emp SET sal=sal-1000 WHERE CURRENT OF emp_cur;
END LOOP;
COMMIT;
END;
--我們可以看到游標(biāo)FOR循環(huán)確實很好的簡化了游標(biāo)的開發(fā),我們不在需要open、fetch和close語句,不在需要用%FOUND屬性檢測是否到最后一條記錄,這一切Oracle隱式的幫我們完成了。
--給經(jīng)理加薪5000,其他加薪1000
DECLARE
CURSOR emp_cur IS
SELECT * FROM emp FOR UPDATE;
BEGIN
FOR emp_row IN emp_cur
LOOP
IF emp_row.job='MANAGER' THEN
UPDATE emp SET sal=sal+5000 WHERE CURRENT OF emp_cur;
ELSE
UPDATE emp SET sal=sal+1000 WHERE CURRENT OF emp_cur;
END IF;
END LOOP;
END;
下面給大家介紹oracle游標(biāo)cursor簡單使用
總共介紹兩種游標(biāo)一種高效使用游標(biāo)cursor 、sys_refcursor 、 bulk collect
1、cursor游標(biāo)使用
/*簡單cursor游標(biāo)
*students表里面有name字段,你可以換做其他表測試
*/
--定義
declare
--定義游標(biāo)并且賦值(is 不能和cursor分開使用)
cursor stus_cur is select * from students;
--定義rowtype
cur_stu students%rowtype;
/*開始執(zhí)行*/
begin
--開啟游標(biāo)
open stus_cur;
--loop循環(huán)
loop
--循環(huán)條件
exit when stus_cur%notfound;
--游標(biāo)值賦值到rowtype
fetch stus_cur into cur_stu;
--輸出
dbms_output.put_line(cur_stu.name);
--結(jié)束循環(huán)
end loop;
--關(guān)閉游標(biāo)
close stus_cur;
/*結(jié)束執(zhí)行*/
end;
執(zhí)行結(jié)果
SQL> declare
--定義游標(biāo)并且賦值(is 不能和cursor分開使用)
cursor stus_cur is select * from students;
--定義rowtype
cur_stu students%rowtype;
/*開始執(zhí)行*/
begin
--開啟游標(biāo)
open stus_cur;
--loop循環(huán)
loop
--循環(huán)條件
exit when stus_cur%notfound;
--游標(biāo)值賦值到rowtype
fetch stus_cur into cur_stu;
--輸出
dbms_output.put_line(cur_stu.name);
--結(jié)束循環(huán)
end loop;
--關(guān)閉游標(biāo)
close stus_cur;
/*結(jié)束執(zhí)行*/
end;
/
楊過
郭靖
付政委
劉自飛
江風(fēng)
任我行
任盈盈
令狐沖
韋一笑
張無忌
朵兒
謝遜
小龍女
歐陽鋒
歐陽鋒
2、sys_refcursor游標(biāo)使用
/*
*游標(biāo)名:sys_refcursor
*特別注意賦值方式:for
*與上重復(fù)內(nèi)容不在敘述
*/
declare
stu_cur sys_refcursor;
stuone students%rowtype;
begin
--這句賦值方式for
open stu_cur for select * from students;
--fetch賦值給rowtype
fetch stu_cur into stuone;
loop
dbms_output.put_line(stuone.name||' '||stuone.hobby);
fetch stu_cur into stuone;
exit when stu_cur%notfound;
end loop;
end;
執(zhí)行結(jié)果
SQL> /*
*游標(biāo)名:sys_refcursor
*特別注意賦值方式:for
*與上重復(fù)內(nèi)容不在敘述
*/
declare
stu_cur sys_refcursor;
stuone students%rowtype;
begin
--這句賦值方式for
open stu_cur for select * from students;
--fetch賦值給rowtype
fetch stu_cur into stuone;
loop
dbms_output.put_line(stuone.name||' '||stuone.hobby);
fetch stu_cur into stuone;
exit when stu_cur%notfound;
end loop;
end;
/
楊過 保護小龍女
郭靖 修煉降龍十八掌
付政委 看小人書
劉自飛 編程寫代碼
江風(fēng) 編程寫代碼
任我行 修煉神功
任盈盈 游山玩水
令狐沖 行俠仗義
韋一笑 吸拾人雪
張無忌 修行
朵兒 洗浴
謝遜 畢生研究屠龍刀
小龍女 修煉玉女心經(jīng)
歐陽鋒 看小人書
補充一種循環(huán)條件
declare
stu_cur sys_refcursor;
stuone students%rowtype;
begin
open stu_cur for select * from students;
fetch stu_cur into stuone;
--特別注意循環(huán)條件的改變
--這個條件是發(fā)現(xiàn)了在循環(huán)
--與上一個notfound不同的
while stu_cur%found loop
dbms_output.put_line(stuone.name||' '||stuone.hobby);
fetch stu_cur into stuone;
end loop;
end;
--普通的fetch into
/*普通方式*/
declare
cursor myemp_cur is select * from myemp;
v_myemp myemp%rowtype;
begin
open myemp_cur;
fetch myemp_cur into v_myemp;
while myemp_cur%found loop
dbms_output.put_line(v_myemp.ename);
fetch myemp_cur into v_myemp;
end loop;
end;
--高效的bulk collect
/*高效bulk collect for*/
declare
cursor myemp_cur
is select * from myemp;
type myemp_tab is table of myemp%rowtype;
myemp_rd myemp_tab;
begin
open myemp_cur;
loop
fetch myemp_cur bulk collect into myemp_rd limit 20;
for i in 1..myemp_rd.count loop
dbms_output.put_line('姓名:'||myemp_rd(i).ename);
end loop;
exit when myemp_cur%notfound;
end loop;
end;
您可能感興趣的文章:- Oracle中觸發(fā)器示例詳解
- 詳解oracle中通過觸發(fā)器記錄每個語句影響總行數(shù)
- Oracle觸發(fā)器trigger詳解
- Oracle觸發(fā)器用法實例詳解
- oracle監(jiān)控某表變動觸發(fā)器例子(監(jiān)控增,刪,改)
- Oracle創(chuàng)建主鍵自增表(sql語句實現(xiàn))及觸發(fā)器應(yīng)用
- Oracle中游標(biāo)Cursor基本用法詳解
- Oracle存儲過程游標(biāo)用法分析
- 快速學(xué)習(xí)Oracle觸發(fā)器和游標(biāo)