Oracle 中 for update 和 for update nowait 的区别
?
原文出處http://bijian1013.iteye.com/blog/1895412
?
一.for update 和 for update nowait 的區(qū)別
首先一點,如果只是select 的話,Oracle是不會加任何鎖的,也就是Oracle對 select 讀到的數(shù)據(jù)不會有任何限制,雖然這時候有可能另外一個進(jìn)程正在修改表中的數(shù)據(jù),并且修改的結(jié)果可能影響到你目前select語句的結(jié)果,但是因為沒有鎖,所以select結(jié)果為當(dāng)前時刻表中記錄的狀態(tài)。
如果加入了for update,則Oracle一旦發(fā)現(xiàn)(符合查詢條件的)這批數(shù)據(jù)正在被修改,則不會發(fā)出該select語句查詢,直到數(shù)據(jù)被修改結(jié)束(被commit),馬上自動執(zhí)行這個select語句。
同樣,如果該查詢語句發(fā)出后,有人需要修改這批數(shù)據(jù)(中的一條或幾條),它也必須等到查詢結(jié)束后(commit)后,才能修改。
for update nowait和 for update 都會對所查詢到得結(jié)果集進(jìn)行加鎖,所不同的是,如果另外一個線程正在修改結(jié)果集中的數(shù)據(jù),for update nowait 不會進(jìn)行資源等待,只要發(fā)現(xiàn)結(jié)果集中有些數(shù)據(jù)被加鎖,立刻返回 “ORA-00054錯誤,內(nèi)容是資源正忙, 但指定以 NOWAIT 方式獲取資源”。
for update 和 for update nowait 加上的是一個行級鎖,也就是只有符合where條件的數(shù)據(jù)被加鎖。如果僅僅用update語句來更改數(shù)據(jù)時,可能會因為加不上鎖而沒有響應(yīng)地、莫名其妙地等待,但如果在此之前,for update NOWAIT語句將要更改的數(shù)據(jù)試探性地加鎖,就可以通過立即返回的錯誤提示而明白其中的道理,或許這就是For Update和NOWAIT的意義之所在。
經(jīng)過測試,以for update 或 for update nowait方式進(jìn)行查詢加鎖,在select的結(jié)果集中,只要有任何一個記錄在加鎖,則整個結(jié)果集都在等待系統(tǒng)資源(如果是nowait,則拋出相應(yīng)的異常)
二.for update nowait 與 for update 的目的
鎖定表的所有行,排斥其他針對這個表的寫操作。確保只有當(dāng)前事務(wù)對指定表進(jìn)行寫操作。
for update nowait和 for update的區(qū)別:
別的事務(wù)要對這個表進(jìn)行寫操作時,是等待一段時間還是馬上就被數(shù)據(jù)庫系統(tǒng)拒絕而返回。制定采用nowait方式來進(jìn)行檢索,所以當(dāng)發(fā)現(xiàn)數(shù)據(jù)被別的session鎖定中的時候,就會迅速返回ORA-00054錯誤,內(nèi)容是資源正忙。所以在程序中我們可以采用nowait方式迅速判斷當(dāng)前數(shù)據(jù)是否被鎖定中,如果鎖定中的話,就要采取相應(yīng)的業(yè)務(wù)措施進(jìn)行處理。
實例:
打開一個PL/SQL,執(zhí)行update deployop.tt t set t.a = 'X' where t.a = '1';
再打開另一個PL/SQL,執(zhí)行select * from tt t where t.a in ('1','2') for update nowait; 返回RA-00054錯誤,內(nèi)容是資源正忙。
如在上面窗口中執(zhí)行select * from tt t where t.a in ('1','2') for update,結(jié)果只是阻塞,不返回錯誤,如下所示:
三.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)用程序中對鎖的等待時間進(jìn)行更多的控制
3.對于交互式應(yīng)用程序非常有用,因為這些用戶不能等待不確定
4.若使用了skip locked,則可以越過鎖定的行,不會報告由wait n 引發(fā)的"資源忙"異常報告
實例:
在plsql develope中打開兩個sql窗口
在1窗口中運(yùn)行如下sql
select * from t where a='1' for update;
在2窗口中運(yùn)行sql1
sql1:select * from t where a='1'; 這一點問題也沒有,因為行級鎖不會影響純粹的select語句
再運(yùn)行sql2
sql2:select * from t where a='1' for update; 則這一句sql在執(zhí)行時,永遠(yuǎn)處于等待狀態(tài),除非窗口1中sql被提交或回滾。
如何才能讓sql2不等待或等待指定的時間呢? 我們再運(yùn)行sql3
sql3: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
sql4:select * from t where a='1' for update nowait skip Locked; 則執(zhí)行sql時,即不等待,也不報資源忙異常,略過鎖定的行,只顯示未鎖定的行
例:
窗口1:
窗口2:略過鎖定的a=1的行,只顯示未鎖定的a=2的行
?
四.select for update of
這個of子句在牽連到多個表時,具有較大作用,如不使用of指定鎖定的表的列,則所有表的相關(guān)行均被鎖定,若在of中指定了需修改的列,則只有與這些列相關(guān)的表的行才會被鎖定。
實例1:
select * from tt,tt2 where tt.a = tt2.a2 for update;對二張表都進(jìn)行了整表鎖定
select * from tt2 for update wait 3;試圖鎖定tt2表,3秒后拋出“ORA-30006: 資源已被占用;執(zhí)行操作時出現(xiàn)WAIT 超時”的錯誤
實例2:
select * from tt,tt2 where tt.a = tt2.a2 for update of a;只鎖定tt表,對tt2表不鎖定
select * from tt2 for update wait 3;成功鎖定了tt2表
select * from tt for update wait 3;試圖鎖定tt表,3秒后拋出“ORA-30006: 資源已被占用;執(zhí)行操作時出現(xiàn)WAIT 超時”的錯誤
可以得到結(jié)論,for update of columns 用在多表連接鎖定時,可以指定要鎖定的是哪幾張表,而如果表中的列沒有在for update of 后面出現(xiàn)的話,就意味著這張表其實并沒有被鎖定,其他用戶是可以對這些表的數(shù)據(jù)進(jìn)行update操作的。這種情況經(jīng)常會出現(xiàn)在用戶對帶有連接查詢的視圖進(jìn)行操作場景下。用戶只鎖定相關(guān)表的數(shù)據(jù),其他用戶仍然可以對視圖中其他原始表的數(shù)據(jù)來進(jìn)行操作。
小結(jié):
?
轉(zhuǎn)載于:https://www.cnblogs.com/MuyouSome/p/3964276.html
總結(jié)
以上是生活随笔為你收集整理的Oracle 中 for update 和 for update nowait 的区别的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: FileSystemWatcher监听文
- 下一篇: 记一次意外