【后端开发】(含图解与实例)乐观锁、悲观锁和分布式锁,做项目时到底该怎么选?

前言

乐观锁、悲观锁、分布式锁,几乎是后端面试和项目实践里绕不开的三个概念。如果只看定义,它们似乎并不难理解:
乐观锁:假设并发冲突不严重,更新数据时再校验是否被别人改过;
悲观锁:假设并发冲突一定会发生,操作前先加锁,再执行后续逻辑;
分布式锁:在多服务节点部署的场景下,保证同一时刻只有一个节点能执行某段逻辑。
很多人背到这里,就觉得自己已经掌握了,但真正到了面试追问或者项目落地时,问题往往就开始了:

  • 乐观锁和悲观锁到底是在锁什么?
  • 分布式锁和前两者是同一类问题吗?
  • 库存扣减该用哪个?
  • 定时任务重复执行又该用哪个?
  • 分布式锁能不能替代乐观锁、悲观锁?

1 乐观锁:适合冲突少的更新场景

很多人第一次听到“乐观锁”时,会下意识觉得它是一种“真的加了锁”的机制。其实不是,乐观锁不强调“先锁住”,而强调“更新时确认数据还是不是我看到的那份数据”。

1.1 乐观锁到底在解决什么问题?

在并发场景下,多个请求可能同时修改同一份数据,如果没有任何控制,就容易出现覆盖更新的问题。
比如,商品库存原本是10:

  • 请求A读到库存是10
  • 请求B也读到库存是10
  • A扣减后写回9
  • B也扣减后写回9
    看起来两个请求都成功了,但实际上库存只少了1次,这就是典型的并发更新问题,导致数据的不一致性
    image

1.2 乐观锁通常怎么实现?

最常见的做法,是给数据加一个version版本号字段。
比如一条商品记录长这样:

id = 1   stock = 10   version = 3

当某个请求读到这条数据时,它不仅会读到库存,也会读到当前版本号。后续更新时,不是直接根据id去更新,而是带着这个version一起更新:

update product
set stock = stock - 1,
    version = version + 1
where id = 1
  and version = 3;

这条 SQL 的关键点不在stock - 1,而在:

and version = 3

只有当这条数据的版本号还是我最初读到的 3 时,我这次更新才允许成功。
如果更新成功,说明这期间没人动过这条数据。
如果更新失败,说明别的请求已经先一步改过了,你当前这次操作所基于的数据已经过期。

1.3 乐观锁实际例子

假设现在有一个“普通商品扣库存”的场景,库存初始值为

stock = 10    version = 5

这时请求A和请求B同时进来。
第一步,请求A和请求B都查询到

stock = 10    version = 5

然后两个请求都准备执行更新。
第二步,请求 A 先执行:

update product
set stock = 9,
    version = 6
where id = 1
  and version = 5;

执行成功,数据库中数据变成:

stock = 9    version = 6

第三步,接着请求 B 再执行:

update product
set stock = 9,
    version = 6
where id = 1
  and version = 5;

这时候就会失败,因为数据库中的version已经不是5,而是6了。
这就是乐观锁的本质:不是抢着先锁住资源,而是通过版本一致性校验,拒绝基于旧数据的更新。
image

2 悲观锁:适合高冲突、强一致场景

如果说乐观锁的思路是“先假设冲突不常发生”,
那悲观锁正好相反:它先假设冲突一定会发生,所以干脆在操作前先把资源锁住。
悲观锁的重点不是“更新失败后怎么补救”,而是在真正修改数据之前,就先阻止别人同时改。

2.1 悲观锁到底在解决什么问题?

悲观锁解决的依然是并发场景下对同一份数据的竞争更新问题。
但它和乐观锁的处理方式完全不同。
乐观锁是

  • 允许大家先一起读
  • 一起尝试更新
  • 最后根据版本号判断谁成功、谁失败

悲观锁是

  • 谁先拿到锁,谁先操作
  • 没拿到锁的人先等着
  • 等前一个操作完成后,后面的人再继续

所以从结果上看,悲观锁更像是在并发场景下,把对关键数据的修改强行串行化。

可以用一个账户余额扣减的例子来解释悲观锁。
假设一个账户当前余额是100元,现在同时来了两个扣款请求:

  • 请求A:扣80元
  • 请求B:扣30元

如果没有做好并发控制,就可能出现问题:

  • A读到余额100
  • B也读到余额100
  • A判断余额足够,扣减成功
  • B也判断余额足够,扣减成功

结果两个请求都通过了校验,总共扣了110元,账户上没有那么多钱,这样账户就出问题了。
image

2.2 悲观锁通常怎么实现?

在后端开发里,最常见的悲观锁实现,就是数据库的行锁

select * from account
where id = 1
for update;

这条SQL的核心是最后的:

for update

它表示:在当前事务提交之前,把这条记录锁住。后续如果其他事务也想对这条记录加锁或修改这条记录,就必须等待当前事务释放锁。

2.3 悲观锁实际例子

还是刚刚提到的余额扣减例子,如果应用悲观锁的话,会变成下面这样:
第一步,请求A先开启事务并加锁:

select balance from account
where id = 1
for update;

数据库把这条账户记录锁住。
第二步,请求B也来了,它如果也想对同一条账户记录执行for update,就会被阻塞,必须等待请求A提交事务。
第三步,请求A在事务里完成校验和更新,

  • 查到balance = 100
  • 判断100 >= 80,允许扣款
  • 执行更新,余额变成20
  • 提交事务
  • 释放锁

第四步,请求A释放锁后,请求B才能获得锁,才能进行查询操作,这时它再查余额,读到的已经是最新值20,然后会发现20 < 30,余额不足,从而扣款失败。
这样一来,系统就不会出现“两个请求都基于旧余额通过校验”的问题。
image

3 分布式锁:适合多节点之间抢执行权

讲到这里,很多人会自然地把分布式锁和乐观锁、悲观锁并列理解,觉得它们只是“不同类型的锁”。

但严格来说,它们并不完全在同一个维度上。
因为前两者主要解决的是:并发更新同一份数据时,如何保证结果正确。而分布式锁更关注的是:在多个服务实例同时运行的情况下,某个动作到底该由谁来执行

分布式锁很多时候锁的不是“某条数据库记录”,而是:

  • 某个任务的执行资格
  • 某段业务逻辑的处理权
  • 某个全局动作的唯一执行者

3.1 为什么单机锁到了分布式场景就不够用了?

在单机应用里,如果你想让一段代码同一时刻只能被一个线程执行,往往可以直接用synchronizedReentrantLock
这类本地锁的前提是,所有竞争者都在同一个进程、同一台机器里
但后端服务一旦真正上线,往往不会只部署一个实例。这时候问题就来了,比如你写了一个定时任务,代码里加了synchronized,你以为这样就能保证“同一时刻只执行一次”。但实际上:

  • 机器A上的线程能进来
  • 机器B上的线程也能进来
  • 它们彼此根本看不到对方的锁
    image

3.2 分布式锁通常怎么实现?

在工程里,分布式锁的实现方式不止一种,但最常见的是Redis分布式锁。
最基础的做法通常类似这样:

SET lock_key unique_value NX PX 30000
  • NX:只有当key不存在时才设置成功,这保证了只有一个实例能抢到锁
  • PX 30000:设置过期时间30秒,避免服务宕机后锁永远不释放
  • unique_value:锁的唯一标识,释放锁时要校验,避免把别人的锁删掉

分布式锁最关键的不是“加锁成功”本身,而是“锁的持有、过期、释放”都要设计清楚,因为它不像本地锁那样天然可靠,分布式锁涉及网络、超时、节点故障,所以边界问题会多很多。

3.3 分布式锁实际例子

用“定时任务只允许一个实例执行”这个经典例子来讲解:
假设系统部署了服务A和服务B两个实例,现在有一个定时任务,每天凌晨扫描超时订单并自动取消
如果没有分布式锁,那么在同一时刻,服务A会触发一次任务,服务B也会触发一次任务。
这样就可能出现:

  • 同一批订单被重复扫描
  • 同一笔退款逻辑被重复执行
  • 同一条通知被重复发送

很多人会说:“那我不是已经写了幂等了吗?”幂等当然很重要,但它解决的是“重复执行后结果尽量不出错”;
而分布式锁解决的是:从源头上减少重复执行,让同一时刻只有一个实例先进去做。他们解决问题的侧重点不一样。
一个典型流程通常是这样:
第一步,服务A和服务B同时触发任务,它们都准备执行“取消超时订单”,然后都去竞争同一把锁,锁名为:

cancel_timeout_order_task_lock

第二步,只有一个实例抢锁成功,假设服务A成功抢到了锁,服务A开始执行业务。
第三步,服务B因为没抢到锁,所以直接退出,或者稍后重试,直到服务A执行完成后释放锁,下一轮任务再重新竞争。
image

4 不同业务中到底怎么选锁?

写到这里,其实最重要的已经不是再去背“乐观锁是什么、悲观锁是什么、分布式锁是什么”了。真正决定你能不能把这类问题讲明白的,是你有没有意识到:这三种锁根本不是同一个维度上的技术点

很多人会把它们并排记忆,然后试图总结成“乐观锁轻一点,悲观锁重一点,分布式锁更高级一点”。
这种记法最容易在面试和项目里出问题,因为它会让你忽略一个关键事实:乐观锁、悲观锁主要解决的是数据竞争,分布式锁主要解决的是执行权竞争

类型 核心解决的问题 锁/控制的对象 常见实现 适合场景 主要优势 主要代价
乐观锁 并发更新同一份数据时,避免基于旧数据覆盖更新 数据版本一致性 version 版本号、CAS、时间戳 读多写少、冲突少的更新场景 并发性能好,不容易阻塞 冲突多时失败率高,需要重试
悲观锁 并发更新同一份关键数据时,保证强互斥和强一致 数据访问权 数据库行锁、select ... for update 写冲突高、强一致要求高的场景 逻辑直观,一致性更稳 阻塞明显,吞吐下降,可能死锁
分布式锁 多个服务节点同时竞争某个业务动作的执行权 某段业务逻辑的执行资格 Redis、ZooKeeper、数据库锁表/唯一约束辅助 定时任务单点执行、全局互斥、避免重复执行 能跨节点控制互斥 实现边界复杂,要处理超时、误删、故障等问题
posted @ 2026-04-25 13:01  IronBro  阅读(78)  评论(3)    收藏  举报