Oracle select for update 用法

SELECT FOR UPDATE 用法

1、SELECT…FOR UPDATE 语法
SELECT … FOR UPDATE [OF column_list][WAIT n|NOWAIT][SKIP LOCKED];
其中:
  OF 子句用于指定即将更新的列,即锁定行上的特定列。
  WAIT 子句指定等待其他用户释放锁的秒数,防止无限等待锁释放。

2、for update 作用
select for update 是为了在查询时,避免其他用户对该表进行插入、修改、删除等操作,用于保证表数据的一致性。
例如:
select * from zh_tb for update; 会等待行锁释放之后,返回查询结果。
select * from zh_tb for update nowait; 不等待行锁释放,提示锁冲突,不返回查询结果。
select * from zh_tb for update wait 5; 等待5秒,若行锁仍未释放,则提示锁冲突,不返回查询结果。
select * from zh_tb for update skip locked; 查询返回查询结果,忽略掉有行锁的记录。
注意:如果查询带条件那么只会锁定条件中的行数据,未带条件则会锁定表。

3、使用 FOR UPDATE WAIT 子句的优点
防止无限期地等待被锁定的行。
允许应用程序中对锁的等待时间进行更多的控制。
对于交互式应用程序非常有用,因为这些用户不能等待不确定。
若使用了skip locked,则可以越过锁定的行,不会报告由wait n 引发的‘资源忙’异常报告。

4、for update 用法
在Oracle中如果只是使用select的话,Oracle是不会给数据加锁的。当数据正在被另一个进程修改的时候,使用select得到的结果就不是最新的,这就需要使用for update,当oracle发现满足条件的记录正在被更新时,不会立即执行select语句,而是一直等待下去,直到更新结束才会执行select语句。

for update 用法举例:
会话1执行如下命令:
SQL> create table zh_tb(c1 varchar(20) ,c2 varchar(20));

Table created.

SQL> select * from zh_tb;

no rows selected

SQL> insert into zh_tb values ('ZZH','ZZH');

1 row created.

SQL> insert into zh_tb values ('AHERN','AHERN');

1 row created.

SQL> select * from zh_tb for update;

C1                   C2
-------------------- --------------------
ZZH                  ZZH
AHERN                AHERN

SQL> commit;

在会话2执行如下查询:
SQL> select * from zh_tb where c1='ZZH' for update;

C1                   C2
-------------------- --------------------
ZZH                  ZZH
此时会话2使用 for update 锁定了查询出来的记录:

再次在会话一执行如下语句查询所有记录:
SQL> select * from zh_tb for update;

发现无法查询出结果,并且会话一直处于等待状态,这是因为会话2里的语句锁住了C1='ZZH'的行,此时会话会一直处于等待状态。只有等会话2中提交了事务之后,会话1才能返回查询结果。
此时我们在会话2执行提交命令提交事务,会话1立即返回了查询结果如下:
SQL> commit;

Commit complete.

SQL> 
提交后发现会话1立即返回了查询结果如下:
SQL> select * from zh_tb for update;
... 

C1                   C2
-------------------- --------------------
ZZH                  ZZH
AHERN                AHERN

SQL> SQL> 

如果大家对 for uodate nowait、for update skip locked 和 for update wair n 用法感兴趣,可以采用如下语句在不同的会话中去执行验证它不同的锁定逻辑。
select * from zh_tb for update nowait;
select * from zh_tb for update wait 6;
select * from zh_tb where rownum<=3 for update skip locked;

5、使用场景
FOR UPDATE 适合可以等待其他事务释放锁定的场景,这样可以逐一处理数据行,确保数据的一致性。
FOR UPDATE NOWAIT 适合那些需要立即知道是否可以锁定所需行的场景,如果无法立即获得锁定,则可以迅速做出响应。
注意:选择使用FOR UPDATE还是FOR UPDATE NOWAIT取决于具体应用场景和对事务等待时间的容忍度。

FOR UPDATE NOWAIT 和 FOR UPDATE的区别:
两者的区别在于如果所选行已经被另一个事务锁定,FOR UPDATE NOWAIT 会立即引发一个 ORA-00054 错误,而不是一直等待其他事务释放锁。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Ahern_

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值