Oracle中select ... for update的用法

本文通过实战演示了 SQL 语句中的 FOR UPDATE 子句及其附加选项 NOWAIT、WAIT 和 SKIP LOCKED 的用法,展示了如何在并发环境中有效管理数据库锁。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

语法:

SELECT... FOR UPDATE [OF column_list][WAIT n|NOWAIT][SKIP LOCKED];

其中OF 子句用于指定即将更新的列,即锁定行上的特定列;WAIT 子句指定等待其他用户释放锁的秒数,防止无限期的等待。

使用“FORUPDATE WAIT”子句的优点如下:

1防止无限期地等待被锁定的行;

2允许应用程序中对锁的等待时间进行更多的控制。

3对于交互式应用程序非常有用,因为这些用户不能等待不确定

4若使用了skip locked,则可以越过锁定的行,不会报告由wait n引发的‘资源忙’异常报告


实验:

createtable t(a varchar2(20),b varchar2(20));

insertinto t values('1','1');

insertinto t values('2','2');

insertinto t values('3','3');

insertinto t values('4','4');

insertinto t values('5','5');

insertinto t values('6','6');


(1)在PLSQL Developer中打开两个窗口,在窗口1中执行

select* from t where a='1' for update;

结果如下

1.jpg

2014-10-8 14:26 上传
下载附件 (24.67 KB)


在窗口2中执行

select* from t where a='1';

结果如下

2.jpg

2014-10-8 14:24 上传
下载附件 (23.48 KB)

可见此时能正常查询。


在窗口2中执行

select* from t where a='1' for update;

发现无法查询出结果且PLSQLDeveloper的执行按钮一直为灰色。

3.jpg

2014-10-8 14:25 上传
下载附件 (68.01 KB)


这是因为表被窗口1里的语句锁住了,窗口2处于等待状态。

只有等窗口1中提交了事务之后才能在窗口2中正常执行上述语句。

在窗口1中点击提交事务的按钮后,窗口2中立马显示出正常结果

把窗口1和2中未提交的事务都提交,以便进行下一步的实验。


(2)

在窗口1中执行select * from t where a='1' for update;

在窗口2中执行select * from t where a='1' for update nowait;

立马报资源正忙的错误:

4.png

2014-10-8 14:26 上传
下载附件 (25.9 KB)


关掉上面的错误提示窗口,在窗口2中执行

select* from t where a='1' for update wait 6;

则6秒之后报错:

5.png

2014-10-8 14:27 上传
下载附件 (27.09 KB)


关掉上面的错误提示,在窗口2中执行

select * from t where a='1' for update skiplocked;

则既不等待,也不报错,也查询不出结果:

6.png

2014-10-8 14:27 上传
下载附件 (54.86 KB)


把窗口1和2中未提交的事务都提交,以便进行下一步的实验。


(3)

在窗口1中执行:

select* from t where rownum<=3 for update skip locked;

结果如下:

7.png

2014-10-8 14:28 上传
下载附件 (63.21 KB)


在窗口2中执行:
select * from t where rownum<=6 for update skip locked;

结果如下:

8.jpg

2014-10-8 14:29 上传
下载附件 (26.25 KB)

可见前三条数据因被窗口1锁住而没有查出来。

转自 http://www.itpub.net/thread-1891268-1-1.html

### 回答1: Oracle中的SELECT FOR UPDATE语句用于锁定查询结果集中的行,以便在事务中进行更新操作。当一个事务使用SELECT FOR UPDATE语句锁定了某些行时,其他事务将无法对这些行进行更新操作,直到锁定的事务释放了锁。这种机制可以确保数据的一致性和完整性。 ### 回答2: Oracleselect for update 是一种锁定行功能,它可以确保在一个事务中,当一个会话开始更新数据时,另一个会话无法修改这些数据。 使用 select for update 首先需要使用 select 语句,该语句从需要更新的表中选择需要锁定的行。然后使用 for update 子句将这些行锁定。例如,下面是一个使用 select for update 的 SQL 语句: SELECT * FROM myTable WHERE id = 1 FOR UPDATE; 这个语句将检索 myTable 表中 id 为 1 的行,并将其锁定,以便其他会话无法修改此行,直到当前会话完成。 请注意,使用 select for update 会对数据库性能产生负面影响。由于锁定行后其他会话无法访问这些数据,因此可能会出现锁定竞争的情况,导致其他会话阻塞等待。因此,应谨慎使用 select for update,只在必要时使用它。同时还需要在程序中考虑一些优化思路,例如使用更细粒度的锁定,减少锁定时间等手段,以降低锁定导致的性能问题。 总之,select for updateOracle 中的一个很有用的功能,它可以确保数据的一致性和完整性。但是,在使用它时,需要仔细考虑并合理利用,以提高其性能效果。 ### 回答3: 在Oracle数据库中,SELECT ... FOR UPDATE语句用于锁定查询结果集中的行,以防止其他事务修改它们。在使用SELECT ... FOR UPDATE语句时,要在语句中明确地指定哪些列需要被锁定。如果不指定任何列,则默认会锁定所有列。 使用SELECT ... FOR UPDATE语句可以保证多个事务在并发修改同一行数据时不会发生冲突。这种锁定行的方式称为行级锁定。 SELECT ... FOR UPDATE语句可以与基于行级锁定的事务并发使用,并且可以充分利用Oracle数据库并发控制机制,提高系统的性能。 例子: 假设有一个账户表,在多个线程中需要对账户进行加款操作,需要保证加款时不会有其他线程同时进行减款操作。 可以使用以下SQL语句来实现这个功能: ```sql SELECT * FROM account WHERE account_number = '123456' FOR UPDATE; ``` 该语句会锁定account_number为"123456"的行,以防止其他事务修改该行数据。当一行被锁定时,其他事务不能修改该行。如果其他事务尝试修改该行,则会被阻塞,直到锁定被释放。 在加款操作完成后,需要手动提交事务或者回滚事务,以释放锁定。如果事务没有被提交或回滚,则锁定会一直存在,直到事务结束。 ```sql UPDATE account SET balance = balance + 100 WHERE account_number = '123456'; ``` 需要注意的是,使用SELECT ... FOR UPDATE语句时要小心死锁的问题。如果多个事务在查询同一行数据时都使用了FOR UPDATE语句,可能会出现死锁情况,这是需要谨慎处理。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值