sql存儲過程動態語句
本文實例講述了MySQL存儲過程中實現執行動態SQL語句的方法。分享給大家供大家參考。具體實現方法如下:
mysql>
mysql>
delimiter
$$
mysql>
mysql>
CREATE
PROCEDURE
set_col_value
->
(in_table
VARCHAR(128),
->
in_column
VARCHAR(128),
->
in_new_value
VARCHAR(1000),
->
in_where
VARCHAR(4000))
->
->
BEGIN
->
DECLARE
l_sql
VARCHAR(4000);
->
SET
l_sql=CONCAT_ws('
',
->
'UPDATE',in_table,
->
'SET',in_column,'=',in_new_value,
->
'
WHERE',in_where);
->
SET
@sql=l_sql;
->
PREPARE
s1
FROM
@sql;
->
EXECUTE
s1;
->
DEALLOCATE
PREPARE
s1;
->
END$$
Query
OK,
0
rows
affected
(0.00
sec)
希望本文所述對大家的MySQL資料庫程序設計有所幫助。
Ⅱ Sql存儲過程 如何動態修改sql語句!
利用動態sql:
將整個sql語句編輯成一個字元串,最終通過execute語句來執行動態sql
Select @Count = count(ID)
FROM NetDIY_Procts_CN
那麼這個呢?需要把結果返回到@Count中,要怎麼在這個sql加上@SqlWhere 條件語句
exec('select ' + @Count + '= count(ID) from NetDIY_Procts_CN ' + @SqlWhere)
SELECT * FROM NetDIY_Procts_CN Where weighting <= @Name order By weighting desc
exec('select * FROM NetDIY_Procts_CN Where weighting <= '+@Name+' ' + @SqlWhere+' order By weighting desc')
兩種都出錯
Ⅲ oracle存儲過程中如何執行動態SQL語句
有時需要在oracle
存儲過程
中執行動態SQL
語句
,例如表名是動態的,或欄位是動態的,或查詢命令是動態的,可用下面的方法:
set
serveroutput
on
declare
n
number;
sql_stmt
varchar2(50);
t
varchar2(20);
begin
execute
immediate
'alter
session
set
nls_date_format=''YYYYMMDD''';
t
:=
't_'
||
sysdate;
sql_stmt
:=
'select
count(*)
from
'
||
t;
execute
immediate
sql_stmt
into
n;
dbms_output.put_line('The
number
of
rows
of
'
||
t
||
'
is
'
||
n);
end;
如果動態SQL
語句
很長很復雜,則可用包裝.
CREATE
OR
REPLACE
PACKAGE
test_pkg
IS
TYPE
cur_typ
IS
REF
CURSOR;
PROCEDURE
test_proc
(v_table
VARCHAR2,t_cur
OUT
cur_typ);
END;
/
CREATE
OR
REPLACE
PACKAGE
BODY
test_pkg
IS
PROCEDURE
test_proc
(v_table
VARCHAR2,t_cur
OUT
cur_typ)
IS
sqlstr
VARCHAR2(2000);
BEGIN
sqlstr
:=
'SELECT
*
FROM
'||v_table;
OPEN
t_cur
FOR
sqlstr;
END;
END;
/
在oracle
中批量導入,導出和刪除表名以某些字元開頭的表
spool
c:\a.sql
select
'drop
table
'
||
tname
||
';'
from
tab
where
tname
like
'T%';
spool
off
@c:\a
Ⅳ 動態SQL語句與存儲過程,誰有高見謝謝
如果你不寫存儲過程你有沒想過會效率根低呢!
說白了你就是不明白存儲過程!
在一個名稱下存儲並作為一個單元進行編譯和處理的T-SQL 語句的集合。
在一個存儲過程內,可以設計、編碼、測試和執行某個常用任務所需的 SQL 語句和邏輯。之後,每個需要執行該任務的應用程序只須執行此存儲過程即可。
允許模塊化程序設計
對於需要重復執行的代碼,執行效率更高
減少網路流量
可作為安全機制使用
好了如果你不用!那好我問你SQL注入你要怎麼防止!
而且插入數據如果出現問題!數據會保存么!
這樣你就會有那樣這樣的BAG出現!
存儲過程為什麼叫存儲過程呢!
就是因為它預先把T-SQL語句存起來!在執行的時候系統會省掉很多緩存資源!
而且不會浪費資源!
假如有幾百條的存儲過程要寫!又要計算!
那麼為什麼不可以在客戶端實現一些這樣又可以減輕SQL的負擔!
現在的電腦配置能很高啦!問題一般出現不大!
界面驗證呀!像一些簡單的查詢大可以在客戶端實現么!
敏感數據在用存儲過程計算!
我就做你說的那種第二種的開發!
效果也挺不錯!
問題不在於存儲過程!在於你代碼的如何編寫!
怎麼樣能夠大大提高運行的效率!
例如不必要浪費的常量!開辟會占堆中資源!
有的代碼運行次數太少!可以簡寫!甚至不寫!非必要都可以省略!
4層架構知道吧!盡量的做到代碼要重用!可大大減少資源的浪費!
SQL的負擔也會騰出來許多!
扯遠啦!
好了就這吧!
希望對你有幫助!
Ⅳ 存儲過程里,如何調用動態子存儲過程
您好,很高興為您解答。
/*動態執行存儲過程DEMO*/
DECLARE
v_Procere_Name VARCHAR2(320); --存儲過程名
v_Input_Parameter1 VARCHAR2(320); --傳入參數1
v_Input_Parameter2 VARCHAR2(320); --傳入參數2
v_Return_Int INTEGER; --接收返回值
v_Return_String VARCHAR2(320); --接收返回值
v_Sql_Str VARCHAR2(320); --動態執行SQL語句
BEGIN
/*獲取存儲過程名,可以從資料庫中動態讀取需要執行的存儲過程,本DEMO直接賦值*/
v_Procere_Name := 'Wms_Application_Out_p.Application_Out_List_Refuse(:i_Str1,:i_Str2,_Return_Int,_Return_String)';
/*本DEMO假定需要調用的存儲過程有兩個傳入參數和兩個傳出參數*/
v_Input_Parameter1 := 'test01'; --傳入參數1,在實際應用中根據具體情況機型賦值
v_Input_Parameter2 := 'test01'; --傳入參數2,在實際應用中根據具體情況機型賦值
/*動態生成需要執行存儲過程的SQL語句*/
v_Sql_Str := 'BEGIN ' || v_Procere_Name || '; END;';
/*執行該SQL語句*/
EXECUTE IMMEDIATE v_Sql_Str
USING IN v_Input_Parameter1, IN v_Input_Parameter2, OUT v_Return_Int, OUT v_Return_String; --參數類型和傳遞順序必須與存儲過程中的保持一致
/*輸出返回值*/
Dbms_Output.Put_Line(v_Return_Int || v_Return_String);
END;
/*動態執行函數DEMO*/
DECLARE
v_Function_Name VARCHAR2(320); --函數名
v_Input_Parameter1 VARCHAR2(320); --傳入參數1
v_Return_String VARCHAR2(320); --接收返回值
v_Sql_Str VARCHAR2(320); --動態執行SQL語句
BEGIN
/*獲取函數名,可以從資料庫中動態讀取需要執行的函數,本DEMO直接賦值*/
v_Function_Name := 'Return_Str_f(:i_Str1)';
/*本DEMO假定需要調用的函數有1個傳入參數*/
v_Input_Parameter1 := 'test01'; --傳入參數1,在實際應用中根據具體情況機型賦值
/*動態生成需要執行函數的SQL語句*/
v_Sql_Str := 'begin :v_Return_String := ' || v_Function_Name || '; end;';
/*執行該SQL語句*/
EXECUTE IMMEDIATE v_Sql_Str
USING OUT v_Return_String, IN v_Input_Parameter1; --接收返回值必須在前,傳入參數類型和傳遞順序必須與函數中保持一致
/*輸出返回值*/
Dbms_Output.Put_Line(v_Return_String);
END;
/*所調用函數*/
CREATE OR REPLACE FUNCTION Return_Str_f(i_Str1 VARCHAR2) RETURN VARCHAR2 IS
v_Sql_Str VARCHAR2(320);
v_Str1 VARCHAR2(320);
BEGIN
/*生成動態執行SQL*/
v_Sql_Str := 'SELECT ''' || i_Str1 || ''' FROM DUAL';
EXECUTE IMMEDIATE v_Sql_Str
INTO v_Str1; --接收返回值
/*輸出返回結果*/
Dbms_Output.Put_Line('v_str1=' || v_Str1);
RETURN v_Str1;
END;
相關資料:
Oracle PL/SQL中動態執行SQL EXECUTE IMMEDIATE
在plsql中經常遇到sql語句是在過程中動態生成的,這個時候可以用EXECUTE IMMEDIATE來執行生成的sql語句。轉一個javaeye(貌似他也是轉的,找不到出處)上的EXECUTE IMMEDIATE用法。
EXECUTE IMMEDIATE代替了以前Oracle8i中DBMS_SQL package包.它解析並馬上執行動態的SQL語句或非運行時創建的PL/SQL塊.動態創建和執行SQL語句性能超前,EXECUTE IMMEDIATE的目標在於減小企業費用並獲得較高的性能,較之以前它相當輕易編碼.盡管DBMS_SQL仍然可用,
但是推薦使用EXECUTE IMMEDIATE,因為它獲的收益在包之上。
使用技巧
1. EXECUTE IMMEDIATE將不會提交一個DML事務執行,應該顯式提交
假如通過EXECUTE IMMEDIATE處理DML命令,那麼在完成以前需要顯式提交或者作為EXECUTE IMMEDIATE自己的一部分. 假如通過EXECUTE IMMEDIATE處理DDL命令,它提交所有以前改變的數據
2. 不支持返回多行的查詢,這種交互將用臨時表來存儲記錄(參照例子如下)或者用REF cursors.
3. 當執行SQL語句時,不要用分號,當執行PL/SQL塊時,在其尾部用分號.
4. 在Oracle手冊中,未具體覆蓋這些功能。下面的例子展示了所有用到Execute immediate的可能方面.希望能給你帶來方便.
5. 對於Forms開發者,當在PL/SQL 8.0.6.3.版本中,Forms 6i不能使用此功能.
EXECUTE IMMEDIATE用法例子
1. 在PL/SQL運行DDL語句
begin
execute immediate 'set role all';
end;
2. 給動態語句傳值(USING 子句)
declare
l_depnam varchar2(20) := 'testing';
l_locvarchar2(10) := 'Du';
begin
execute immediate 'insert into dept values(:1, :2, :3)'
using 50, l_depnam, l_loc;
commit;
end;
3. 從動態語句檢索值(INTO子句)
declare
l_cntvarchar2(20);
begin
execute immediate 'select count(1) from emp'
into l_cnt;
dbms_output.put_line(l_cnt);
end;
4. 動態調用常式.常式中用到的綁定變數參數必須指定參數類型.黓認為IN類型,其它類型必須顯式指定
declare
l_routin varchar2(100) := 'gen2161.get_rowcnt';
l_tblnam varchar2(20) := 'emp';
l_cntnumber;
l_status varchar2(200);
begin
execute immediate 'begin ' l_routin '(:2, :3, :4); end;'
using in l_tblnam, out l_cnt, in out l_status;
if l_status != 'OK' then
dbms_output.put_line('error');
end if;
end;
5. 將返回值傳遞到PL/SQL記錄類型;同樣也可用%rowtype變數
declare
type empdtlrec is record (empnonumber(4),
enamevarchar2(20),
deptnonumber(2));
empdtl empdtlrec;
begin
execute immediate 'select empno, ename, deptno '
'from emp where empno = 7934'
into empdtl;
end;
6. 傳遞並檢索值.INTO子句用在USING子句前
declare
l_deptpls_integer := 20;
l_nam varchar2(20);
l_loc varchar2(20);
begin
execute immediate 'select dname, loc from dept where deptno = :1'
into l_nam, l_loc
using l_dept ;
end;
7. 多行查詢選項.對此選項用insert語句填充臨時表,用臨時表進行進一步的處理,也可以用REF cursors糾正此缺憾.
declare
l_sal pls_integer := 2000;
begin
execute immediate 'insert into temp(empno, ename) '
'select empno, ename from emp '
'wheresal :1'
using l_sal;
commit;
end;
對於處理動態語句,EXECUTE IMMEDIATE比以前可能用到的更輕易並且更高效.當意圖執行動態語句時,適當地處理異常更加重要.應該關注於捕捉所有可能的異常.
如若滿意,請點擊右側【採納答案】,如若還有問題,請點擊【追問】
希望我的回答對您有所幫助,望採納!
Ⅵ 如何在oracle存儲過程中執行動態sql語句
時需要在oracle 存儲過程中執行動態SQL 語句 ,例如表名是動態的,或欄位是動態的,
或查詢命令是動態的,可用下面的方法:
set serveroutput ondeclaren number;sql_stmt varchar2(50);
t varchar2(20);beginexecute immediate 'alter session set nls_date_format=''YYYYMMDD''';
t := 't_' || sysdate;
sql_stmt := 'select count(*) from ' || t;
execute immediate sql_stmt into n;
dbms_output.put_line('The number of rows of ' || t || ' is ' || n);end;
如果動態SQL 語句 很長很復雜,則可用包裝.
CREATE OR REPLACE PACKAGE test_pkgISTYPE cur_typ IS REF CURSOR;
PROCEDURE test_proc (v_table VARCHAR2,t_cur OUT cur_typ);END;/
CREATE OR REPLACE PACKAGE BODY test_pkgISPROCEDURE test_proc (v_table VARCHAR2,t_cur OUT cur_typ)ISsqlstr VARCHAR2(2000);BEGINsqlstr := 'SELECT * FROM '||v_table;
OPEN t_cur FOR sqlstr;END;END;/
在oracle 中批量導入,導出和刪除表名以某些字元開頭的表
spool c:\a.sql
Ⅶ oracle存儲過程中如何執行動態SQL語句 詳細
有時需要在oracle
存儲過程中執行動態SQL
語句
,例如表名是動態的,或欄位是動態的,
或查詢命令是動態的,可用下面的方法:
set
serveroutput
ondeclaren
number;sql_stmt
varchar2(50);
t
varchar2(20);beginexecute
immediate
'alter
session
set
nls_date_format=''YYYYMMDD''';
t
:=
't_'
||
sysdate;
sql_stmt
:=
'select
count(*)
from
'
||
t;
execute
immediate
sql_stmt
into
n;
dbms_output.put_line('The
number
of
rows
of
'
||
t
||
'
is
'
||
n);end;
如果動態SQL
語句
很長很復雜,則可用包裝.
CREATE
OR
REPLACE
PACKAGE
test_pkgISTYPE
cur_typ
IS
REF
CURSOR;
PROCEDURE
test_proc
(v_table
VARCHAR2,t_cur
OUT
cur_typ);END;/
CREATE
OR
REPLACE
PACKAGE
BODY
test_pkgISPROCEDURE
test_proc
(v_table
VARCHAR2,t_cur
OUT
cur_typ)ISsqlstr
VARCHAR2(2000);BEGINsqlstr
:=
'SELECT
*
FROM
'||v_table;
OPEN
t_cur
FOR
sqlstr;END;END;/
在oracle
中批量導入,導出和刪除表名以某些字元開頭的表
spool
c:\a.sql
Ⅷ 存儲過程含有動態SQL,與靜態SQL有什麼差別
SQL 語句從編譯和運行的角度可以分為兩種,靜態 SQL和 動態 SQL,這兩種 SQL 在使用方式、運行機制和性能表現等方面各有特點 :
靜態 SQL:靜態 SQL 語句一般用於嵌入式 SQL 應用中,在程序運行前,SQL 語句必須是確定的,例如 SQL 語句中涉及的列名和表名必須是存在的。靜態 SQL 語句的編譯是在應用程序運行前進行的,編譯的結果會存儲在資料庫內部。而後程序運行時,資料庫將直接執行編譯好的 SQL 語句,降低運行時的開銷。
動態 SQL:動態 SQL 語句是在應用程序運行時被編譯和執行的,例如,使用 DB2 的互動式工具 CLP 訪問資料庫時,用戶輸入的 SQL 語句是不確定的,因此 SQL 語句只能被動態地編譯。動態 SQL 的應用較多,常見的 CLI 和 JDBC 應用程序都使用動態 SQL。
靜態sql:語句類型在編程時候必須是確定好的。比如
select * from employee where empno='abc'select * from employee where empno='12'都必須是確定的,唯一可以變化的是abc的值。
動態sql:語句類型可以在運行期間指定,比如clp就是最典型的動態sql程序,你可以輸入任何命令。
靜態sql的存取路徑是在運行前就確定好的,而動態sql的存取路徑是在運行時動態生成的。因此生成的存取計劃相對更優,但考慮到生成存取路徑的開銷,有可能應用程序的運行時間相對會比靜態sql長些。