MySQL中for update的作用和用法

本文详细介绍了MySQL的forupdate锁机制,它是行级锁的一种,用于在事务中锁定查询到的行,防止其他事务进行更新操作。forupdate在主键明确时执行行锁,否则可能执行表锁。适用场景包括需要确保数据一致性的业务操作,如火车票订票系统。同时,forupdate也是悲观锁的体现,与乐观锁形成对比。在实际应用中,需谨慎使用以避免死锁问题。

一、for update定义

for update是一种行级锁,又叫排它锁

一旦用户对某个行施加了行级加锁,则该用户可以查询也可以更新被加锁的数据行,其它用户只能查询但不能更新被加锁的数据行。

如果其它用户想更新该表中的数据行,则也必须对该表施加行级锁.即使多个用户对一个表均使用了共享更新,但也不允许两个事务同时对一个表进行更新,真正对表进行更新时,是以独占方式锁表,一直到提交或复原该事务为止。

行锁永远是独占方式锁。

只有当出现如下之一的条件,才会释放共享更新锁:

  1. 执行提交(COMMIT)语句
  2. 退出数据库(LOG OFF)
  3. 程序停止运行

二、概念和用法

通常情况下,select语句是不会对数据加锁,妨碍影响其他的DML和DDL操作。同时,在多版本一致读机制的支持下,select语句也不会被其他类型语句所阻碍。

而select … for update 语句是我们经常使用手工加锁语句。在数据库中执行select … for update ,大家会发现会对数据库中的表或某些行数据进行锁表,在mysql中,如果查询条件带有主键,会锁行数据,如果没有,会锁表。

由于InnoDB预设是Row-Level Lock,所以只有「明确」的指定主键,MySQL才会执行Row lock (只锁住被选取的资料例) ,否则MySQL将会执行Table Lock (将整个资料表单给锁住)。

举个例子: 假设有张表user ,里面有 id 和 name 两列,id是主键。

例1: (明确指定主键,并且数据真实存在,row lock)

-- 开启事务
begin;
SELECT * FROM user WHERE id=3 FOR UPDATE;
SELECT * FROM user WHERE id=3 and name='Tom' FOR UPDATE;
-- 提交事务
commit;

例2: (明确指定主键,但数据不存在,无lock)

begin;
SELECT * FROM user WHERE id=0 FOR UPDATE;
commit;

注意:

上面的说法有误,InnoDB RR级别的锁算法是next-key lock

普通索引无论是否命中都是next-key lock + gap lock

唯一索引并命中则降级为record lock,查不到还是next-key lock + gap lock,所以这里是有锁的,锁定的是一个主键范围,并不是无锁,可以另外开启一个事务,执行下面的脚本,就会发现当上面的事务没有提交时,下面的sql第二句是执行不了的。

begin;
SELECT * FROM user WHERE id = 1 FOR UPDATE;
commit;

例3: (主键不明确,table lock)

begin;
SELECT * FROM user WHERE id<>3 FOR UPDATE;
SELECT * FROM user WHERE id LIKE '%3%' FOR UPDATE;
commit;

例4: (无主键,table lock)

begin;
SELECT * FROM user WHERE name='Tom' FOR UPDATE;
commit;

注意:

  1. FOR UPDATE仅适用于InnoDB,且必须在事务处理模块(BEGIN/COMMIT)中才能生效。
  2. 要测试锁定的状况,可以利用MySQL的Command Mode(命令模式) ,开两个视窗来做测试。
  3. Myisam 只支持表级锁,InnerDB支持行级锁 添加了(行级锁/表级锁)锁的数据不能被其它事务再锁定,也不被其它事务修改。是表级锁时,不管是否查询到记录,都会锁定表。

三、什么时候需要使用for update?

借助for update语句,我们可以在应用程序的层面手工实现数据加锁保护操作。就是那些需要业务层面数据独占时,可以考虑使用for update。

场景上,比如火车票订票,在屏幕上显示有票,而真正进行出票时,需要重新确定一下这个数据没有被其他客户端修改。所以,在这个确认过程中,可以使用for update。

四、for update悲观锁

悲观锁:总是假设最坏的情况,每次去拿数据的时候都认为别人会修改,所以每次在拿数据的时候都会上锁,这样别人想拿这个数据就会阻塞直到它解锁。传统的关系型数据库里边就用到了很多这种锁机制,比如行锁,表锁等,读锁,写锁等,都是在做操作之前先上锁。就像for update,再比如Java里面的同步原语synchronized关键字的实现也是悲观锁。

乐观锁:顾名思义,就是很乐观,每次去拿数据的时候都认为别人不会修改,所以不会上锁,但是在更新的时候会判断一下在此期间别人有没有去更新这个数据,可以使用版本号等机制。乐观锁适用于多读的应用类型,这样可以提高吞吐量,像数据库提供的类似于write_condition机制,其实都是提供的乐观锁。

### MySQLFOR UPDATE 锁机制 `FOR UPDATE` 是 MySQL 提供的一种行级锁定机制,用于在事务中确保数据一致性。当在一个事务中执行 `SELECT ... FOR UPDATE` 查询时,所选中的行会被加上排他锁(exclusive lock),这意味着其他事务无法修改这些被锁定的行,直到当前事务完成并释放锁。 #### 行级锁 vs 表级锁 - 如果查询条件中涉及索引列(尤其是主键或唯一索引),MySQL 通常只会对符合条件的具体行加锁,这种行为称为 **行级锁**[^1]。 - 若查询条件未命中任何索引,则可能导致整个表被锁定,即发生 **表级锁**[^2]。 #### 加锁过程与并发控制 一旦某一行通过 `FOR UPDATE` 被锁定,在另一个事务尝试对该同一行也发起 `FOR UPDATE` 请求时,后者将进入等待状态直至前一事务结束(提交或回滚)。然而需要注意的是,普通的只读型 SELECT 不受影响——即使目标记录已被另一事务以 `FOR UPDATE` 方式锁定,新来的只读请求仍可顺利完成而不需排队等候解锁[^4]。 以下是关于如何运用此功能的一些具体例子: ```sql START TRANSACTION; -- 假设我们想要更新订单号为1的状态之前先确认其现状 SELECT * FROM orders WHERE order_id = 1 FOR UPDATE; UPDATE orders SET status='shipped' WHERE order_id=1; COMMIT; ``` 上述脚本展示了典型的使用模式:启动一个新的显式事务;接着利用带 `FOR UPDATE` 子句的选择命令获取特定条目的最新版本及其上的写权限;最后做出必要的更改再提交变更。 ### 使用场景分析 考虑到性能开销以及潜在死锁风险等因素,合理规划何时何地应用此类强约束显得尤为重要。下面列举了几种适合采用 `FOR UPDATE` 的典型情境: 1. 当应用程序需要基于现有数据库内容计算得出新的数值后再存入DBMS头的时候; 2. 需要防止两个独立进程同时访问相同的数据集从而引发竞争状况的情形下; 3. 实现队列管理或者库存控制系统之类的功能模块期间。 尽管如此,过度依赖于这类悲观策略也可能带来负面影响比如降低吞吐量等负面效应因此应当谨慎权衡利弊之后决定是否采纳该方法论作为解决方案的一部分[^3]. 相关问题
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值