SELECT FOR UPDATE

OracleSelect For Update语句可以实现在读取数据后马上锁定相关资源,防止被其他session修改数据的目的。也就是我们常常谈到的“悲观锁定”(现实应用开发中,使用悲观锁定的情况少之又少,也许是因为乐观锁定的实现更加灵活和便捷的缘故)。
这个小文儿做一个小小的实验,来看看Select For Update语句实现的行级锁定

1.创建实验表table_sfu,并初始化三条数据
sec@ora10g> create table table_sfu (a number);

Table created.

sec@ora10g> insert into table_sfu values (1);

1 row created.

sec@ora10g> insert into table_sfu values (2);

1 row created.

sec@ora10g> insert into table_sfu values (3);

1 row created.

sec@ora10g> commit;

Commit complete.

sec@ora10g> select * from table_sfu;

         A
----------
         1
         2
         3

2.使用Select For Update语句得到第一条数据
sec@ora10g> select * from table_sfu where a = 1 for update;

         A
----------
         1

3.查看一下现在系统中的锁定情况,152会话(即上面语句所在的会话)获得了一个TX锁和一个TM锁了,锁定的表就是TABLE_SFU
sec@ora10g> @lock

lock        lock
holder    holder         lock             lock request       blocked
username  sessid SERIAL# type    id1  id2 mode    mode BLOCK  sessid
-------- ------- ------- ---- ------ ---- ---- ------- ----- -------
SEC          152   14985 TM    15396    0    3       0     0
SEC          152   14985 TX   327722 1790    6       0     0
             164       1 TS        3    1    3       0     0
             165       1 CF        0    0    2       0     0
             165       1 RS       25    1    2       0     0
             165       1 XR        4    0    1       0     0
             166       1 RT        1    0    6       0     0

7 rows selected.

sec@ora10g> col OWNER for a6
sec@ora10g> col OBJECT_NAME for a10
sec@ora10g> select OWNER,OBJECT_NAME,OBJECT_ID,OBJECT_TYPE from dba_objects where object_id = '15396';

OWNER  OBJECT_NAM  OBJECT_ID OBJECT_TYPE
------ ---------- ---------- -------------------
SEC    TABLE_SFU       15396 TABLE

4.另外新打开一个session,执行以下修改任务
sec@ora10g> update table_sfu set a = 100 where a = 1;
OK,效果出现了,这里出现了“锁等待”现象,原因就是因为在第一个session中使用Select For Update语句锁定了第一行数据,不允许其他的session对它修改。

5.这时系统中锁定情况如下,可以看到第一个session(session id是152)会话锁定了第二个session(session id是145)会话的事务
sec@ora10g> @lock

lock        lock
holder    holder         lock             lock request       blocked
username  sessid SERIAL# type    id1  id2 mode    mode BLOCK  sessid
-------- ------- ------- ---- ------ ---- ---- ------- ----- -------
SEC          145   11388 TM    15396    0    3       0     0
SEC          152   14985 TM    15396    0    3       0     0
SEC          152   14985 TX   327722 1790    6       0     1     145
             164       1 TS        3    1    3       0     0
             165       1 CF        0    0    2       0     0
             165       1 RS       25    1    2       0     0
             165       1 XR        4    0    1       0     0
             166       1 RT        1    0    6       0     0

8 rows selected.

6.因为仅仅是锁定了第一条数据,所以其他记录可以顺利的进行修改,如下
sec@ora10g> update table_sfu set a = 200 where a = 2;

1 row updated.

sec@ora10g> commit;

Commit complete.

7.解锁方式:commit或rollback后即完成锁定的接触

8.反过来思考一下,如果Select For Update与要锁定的行已经在其他session中完成了修改,再执行回出现什么效果呢?这个很显然,同样的会出现“锁等待”的现象,不过我想强调的是,这里可以使用nowait和wait选项来进行“探测”待锁定行是否可被锁定
实验效果如下:
第一个session:
sec@ora10g> update table_sfu set a = 100 where a = 1;

1 row updated.

第二个session:
sec@ora10g> select * from table_sfu where a = 1 for update;
此处是“锁等待”效果

sec@ora10g> select * from table_sfu where a = 1 for update nowait;
select * from table_sfu where a = 1 for update nowait
              *
ERROR at line 1:
ORA-00054: resource busy and acquire with NOWAIT specified
这里提示了错误,原因就是已经“探测”到该行已经被别的事务锁定,这里无法对其进行锁定操作。

sec@ora10g> select * from table_sfu where a = 1 for update wait 3;
select * from table_sfu where a = 1 for update wait 3
              *
ERROR at line 1:
ORA-30006: resource busy; acquire with WAIT timeout expired
这里提示的错误内容与上面的一样,不过这里wait 3表示,我等你三秒的时间,如果三秒过后还无法锁定资源,就报错。

9.更进一步,请参考Oracle官方文档中相关的描述
《for_update_clause ::=》
http://download.oracle.com/docs/cd/B19306_01/server.102/b14200/statements_10002.htm#i2126016
语法格式:
FOR UPDATE
[ OF [ [ schema. ]
       { table | view } . ]column
       [, [ [ schema. ]
            { table | view } . ]column
       ]...
]
[ NOWAIT | WAIT integer ]

同上述连接,搜索关键字“for_update_clause”可以得到每个选项的解释信息

《Using the FOR UPDATE Clause: Examples 》
http://download.oracle.com/docs/cd/B19306_01/server.102/b14200/statements_10002.htm#i2130052

10.小结
上面的小实验展示了一下Select For Update行级锁定的效果,Oracle的锁定机制还是非常的灵活的,基于这个锁定可以实现“悲观锁定”。

-- The End --


http://space.itpub.net/519536/viewspace-612048

### SQL中SELECT FOR UPDATE的用法及行为 在SQL中,`SELECT FOR UPDATE` 是一种用于确保数据一致性的机制,通常与事务结合使用。它允许用户在查询某些行时,同时锁定这些行,以防止其他事务对这些行进行修改或删除操作[^4]。 #### 1. 基本语法 `SELECT FOR UPDATE` 的基本语法如下: ```sql SELECT column1, column2, ... FROM table_name WHERE condition FOR UPDATE; ``` - `FOR UPDATE` 表示查询的行会被锁定,直到当前事务结束。 - 锁定的行无法被其他事务修改或删除,但可以被读取。 #### 2. 锁定行为 当执行 `SELECT FOR UPDATE` 时,数据库会对查询结果中的每一行加排他锁(exclusive lock)。这意味着: - 其他事务不能修改或删除这些行,直到当前事务提交或回滚[^5]。 - 其他事务仍然可以读取这些行,但如果尝试修改或删除,将被阻塞,直到锁释放。 #### 3. 示例代码 以下是一个典型的 `SELECT FOR UPDATE` 示例: ```sql BEGIN; -- 查询并锁定指定条件的行 SELECT inventory_count FROM products WHERE product_id = 123 FOR UPDATE; -- 更新库存数量 UPDATE products SET inventory_count = inventory_count - 1 WHERE product_id = 123; COMMIT; ``` 上述代码展示了如何在事务中使用 `SELECT FOR UPDATE` 来确保并发环境下的数据一致性[^6]。 #### 4. 注意事项 - **死锁风险**:如果多个事务同时对同一组行执行 `SELECT FOR UPDATE`,可能会导致死锁。因此,在设计应用程序时需要特别注意事务的顺序和范围[^7]。 - **性能影响**:由于 `FOR UPDATE` 会锁定行,可能导致其他事务等待,从而降低系统吞吐量。应尽量减少锁定的行数和时间。 - **隔离级别**:`SELECT FOR UPDATE` 的行为可能受到数据库隔离级别的影响。例如,在可重复读(Repeatable Read)或串行化(Serializable)隔离级别下,锁定的范围可能更大[^8]。 #### 5. 应用场景 - 银行转账:确保账户余额在更新前不会被其他事务修改。 - 库存管理系统:避免多个用户同时扣减库存导致超卖问题。 - 订单处理:确保订单状态在更新前不会被其他事务修改。 --- ###
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值