掃二維碼與項目經(jīng)理溝通
我們在微信上24小時期待你的聲音
解答本文疑問/技術(shù)咨詢/運營咨詢/技術(shù)建議/互聯(lián)網(wǎng)交流
以下的文章主要是講述Oracle存儲過程中如何正確的使用行鎖的實際示例,如果一張入庫單表中的相關(guān)字段status用來記錄相關(guān)的入庫狀態(tài),入庫前我們要先判斷其入庫的標(biāo)識。如果沒有入庫的話,我們則做入庫的相關(guān)處理。

創(chuàng)新互聯(lián)建站專業(yè)為企業(yè)提供臨川網(wǎng)站建設(shè)、臨川做網(wǎng)站、臨川網(wǎng)站設(shè)計、臨川網(wǎng)站制作等企業(yè)網(wǎng)站建設(shè)、網(wǎng)頁設(shè)計與制作、臨川企業(yè)網(wǎng)站模板建站服務(wù),十余年臨川做網(wǎng)站經(jīng)驗,不只是建網(wǎng)站,更提供有價值的思路和整體網(wǎng)絡(luò)服務(wù)。
否則則做另一個操作,問題來了,當(dāng)兩個人同時做入庫處理時,取得的狀態(tài)都為'未入庫',于是,你可以想到,發(fā)生了兩次入庫操作。在并發(fā)問題中,一不小心就可能出現(xiàn)一些不易發(fā)現(xiàn)的錯誤。這里可以通過一個Oracle存儲過程中使用行鎖來解決這個問題,通過鎖,使該存儲過程不能同時被兩個線程調(diào)用來處理同一條記錄。
- create or replace procedure P1(pdm in varchar2) is
- var_flag char(1);
- begin
- select status into var_flag from #T where dm=pdm for update wait 5;
開始事務(wù)
執(zhí)行業(yè)務(wù)邏輯
修改入庫標(biāo)志
提交事務(wù)
若出異常,回滾
- end P1;
注意:記錄被鎖定之后不可以在該記錄上做操作。
SELECT...FOR UPDATE 語句的語法如下:
- SELECT ... FOR UPDATE [OF column_list][WAIT n|NOWAIT][SKIP LOCKED];
其中:
OF 子句用于指定即將更新的列,即鎖定行上的特定列。
WAIT 子句指定等待其他用戶釋放鎖的秒數(shù),防止無限期的等待。
“使用FOR UPDATE WAIT”子句的優(yōu)點如下:
1防止無限期地等待被鎖定的行;
2允許應(yīng)用程序中對Oracle存儲過程中使用行鎖的等待時間進行更多的控制。
3對于交互式應(yīng)用程序非常有用,因為這些用戶不能等待不確定
4 若使用了skip locked,則可以越過鎖定的行,不會報告由wait n 引發(fā)的‘資源忙’異常報告
示例:
- create table t(a varchar2(20),b varchar2(20));
- insert into t values('1','1');
- insert into t values('2','2');
- insert into t values('3','3');
- insert into t values('4','4');
現(xiàn)在執(zhí)行如下操作:
在plsql develope中打開兩個sql窗口,
在1窗口中運行sql
- select * from t where a='1' for update;
在2窗口中運行sql1q
1.
- select * from t where a='1';
這一點問題也沒有,因為行級鎖不會影響純粹的select語句
再運行sql2
2.
- select * from t where a='1' for update;
則這一句sql在執(zhí)行時,永遠(yuǎn)處于等待狀態(tài),除非窗口1中sql被提交或回滾。
如何才能讓sql2不等待或等待指定的時間呢? 我們再運行sql3
3.
- select * from t where a='1' for update nowait;
則在執(zhí)行此sql時,直接報資源忙的異常。
若執(zhí)行
- select * from t where a='1' for update wait 6;
則在等待6秒后,報 資源忙的異常。
如果我們執(zhí)行sql4
4.
- select * from t where a='1' for update nowait skip Locked;
則執(zhí)行sql時,即不等待,也不報資源忙異常。
現(xiàn)在我們看看執(zhí)行如下操作將會發(fā)生什么呢?
在窗口1中執(zhí)行:
- select * from t where rownum<=3 nowait skip Locked;
在窗口2中執(zhí)行:
- select * from t where rownum<=6 nowait skip Locked;
select for update 也就如此了吧,insert、update、delete操作默認(rèn)加行級鎖,其原理和操作與select for update并無兩樣。
select for update of,這個of子句在牽連到多個表時,具有較大作用,如不使用of指定鎖定的表的列,則所有表的相關(guān)行均被鎖定,若在of中指定了需修改的列,則只有與這些列相關(guān)的表的行才會被Oracle存儲過程中使用行鎖定。

我們在微信上24小時期待你的聲音
解答本文疑問/技術(shù)咨詢/運營咨詢/技術(shù)建議/互聯(lián)網(wǎng)交流