創(chuàng)建存儲(chǔ)過(guò)程
CREATE OR REPLACE PROCEDURE TEST_COUNT AS
v_total number(10);
BEGIN
select count(*) into v_total from testtable;
dbms_output.put_line('總?cè)藬?shù):'||v_total);
END TEST_COUNT;
CREATE OR REPLACE 的意思就是如果這個(gè)存儲(chǔ)過(guò)程沒(méi)有就創(chuàng)建(CREATE ),如果這個(gè)存儲(chǔ)過(guò)程存在,那么就替換(REPLACE)。
v_total number(10)的意思就是聲明一個(gè)變量v_total,它的類型是number(10).
BEGIN關(guān)鍵字,表示開(kāi)始跟 END 存儲(chǔ)過(guò)程的過(guò)程名字是固定的格式。
select count(*) into v_total from testtable;的意思是從testtable行數(shù)綜合和賦值給v_total。
可以換成 select student_name into s_name from testtable,換句話說(shuō)查詢的結(jié)果是可以賦值給上面聲明的變量,可以查詢賦值一個(gè),也可以查詢多個(gè)賦值給多個(gè)變量,當(dāng)然一般不這么做,因?yàn)檫€可以使用游標(biāo)。
使用游標(biāo)的存儲(chǔ)過(guò)程
CREATE OR REPLACE PROCEDURE TEST_COUNT AS
CURSOR test_cursor is select t.id1,t.name from testtable t;
BEGIN
for test_record in test_cursor
loop
dbms_output.put_line(test_record.id1||test_record.name);//打印test_record和test_record.name
end loop;
END TEST_COUNT;
CURSOR test_cursor is select t.id1,t.name from testtable t;
聲明一個(gè)游標(biāo) test_cursor (格式 Cursor XX is),游標(biāo)的意思就類似我們編程語(yǔ)言中的集合,在這里他的結(jié)果就是把查詢testtable中的t.id1和t.name賦值給游標(biāo)test_cursor,接下來(lái)利用 loop循環(huán)(類似for函數(shù))依次遍歷出來(lái)。
帶參數(shù)的存儲(chǔ)過(guò)程
CREATE OR REPLACE PROCEDURE TEST_COUNT(p_id1 in varchar2 default'0') AS
v_name varchar2(20);
BEGIN
select t.name into v_name from testtable t where t.id1=p_id1;
dbms_output.put_line('姓名:'||v_name);
END TEST_COUNT;
要給存儲(chǔ)過(guò)程帶參數(shù)只需要在過(guò)程名后面加括號(hào)帶參數(shù),類似Java(function(param param1))這樣子就可以了,當(dāng)然參數(shù)要聲明類型,帶進(jìn)來(lái)的參數(shù)在后面的存儲(chǔ)過(guò)程中均可以直接使用。
拼接SQL還有使用動(dòng)態(tài)游標(biāo)
接下來(lái)進(jìn)階了,拼接SQL還有使用動(dòng)態(tài)游標(biāo),因?yàn)橹暗挠螛?biāo)都是直接賦值的,不像其他一樣先聲明,后賦值,所以使用動(dòng)態(tài)游標(biāo)可以解決這個(gè)問(wèn)題。
還是剛剛的帶參數(shù)的存儲(chǔ)過(guò)程,如果ID是空的,那我們不是查不到數(shù)據(jù),所以我們可以實(shí)現(xiàn)先判斷id是不是空,如果是空,那么不傳參,如果有才傳參。
CREATE OR REPLACE PROCEDURE TEST_COUNT(p_id1 in varchar2 default'0') AS
type ref_cursor is ref cursor; //類似C語(yǔ)言里面的typedef(個(gè)人見(jiàn)解)
t_result ref_cursor;//所以現(xiàn)在t_result就是一個(gè) 動(dòng)態(tài)游標(biāo)(其實(shí)還是游標(biāo))
t_sql varchar(100);//聲明t_sql來(lái)拼接sql,切記sql在oracle是關(guān)鍵字
v_name varchar(50);
BEGIN
t_sql:='select t.name into v_name from testtable t where 1=1';//1-1是為了把where關(guān)鍵字代入。
if p_id1 is not null
t_sql:=t_sql||'where t_id1='||p_id1;
end if //if .... end if 固定格式
open t_result for t_sql //打開(kāi)動(dòng)態(tài)游標(biāo),其實(shí)就是把結(jié)果集賦值給t_result.
loop
fetch t_result into v_name;
exit when t_result%notfound;//當(dāng)結(jié)果集沒(méi)有什么影響就跳出循環(huán)
dbms_output.put_line(v_name);
end loop;
END TEST_COUNT;
返回結(jié)果集是游標(biāo)
create or replace
package test_package
as
type cursorRef is ref cursor; --定義游標(biāo)引用類型
procedure test_procedure(p_id number,cursor_ref out cursorRef);
end test_package;
第一步創(chuàng)建程序包,其實(shí)一般的存儲(chǔ)過(guò)程都是先創(chuàng)建程序包的。
然后再來(lái)創(chuàng)建存儲(chǔ)過(guò)程。
create or replace
PACKAGE BODY test_package IS
procedure test_procedure(p_id number,cursor_ref out cursorRef)// 因?yàn)槭莖ut的,所以這個(gè)cursor_ref是輸出的也就是返回的游標(biāo),out關(guān)鍵字
BEGIN
open cursor_ref for select * from testtable t where t.id1=p_id;
end test_procedure;
end test_package;
select * from testtable t where t.id1 =1;
這樣子調(diào)用就可以返回游標(biāo)型的結(jié)果集。