当前位置: 首页
编程语言
数据库事务隔离:乐观锁与悲观锁在PHP中的实现

数据库事务隔离:乐观锁与悲观锁在PHP中的实现

热心网友 时间:2026-05-02
转载

数据库事务隔离:乐观锁与悲观锁在PHP中的实现

数据库事务隔离:乐观锁与悲观锁在PHP中的实现

免费影视、动漫、音乐、游戏、小说资源长期稳定更新! 👉 点此立即查看 👈

在Web应用开发中,你有没有遇到过这样的场景:多个用户几乎同时对同一账户进行扣款或修改,结果数据出现了错乱?这背后,其实就是并发控制的问题。要解决它,绕不开两个核心概念:乐观锁和悲观锁。今天,我们就来聊聊它们在PHP中的具体实现方式,看看如何用代码来守护数据的一致性。

一、悲观锁的PHP实现

悲观锁的思路很直接:它默认冲突是常态,所以在动手操作数据之前,就先“占住”它,不让别人碰。这就像去图书馆借一本热门书,管理员会先把它锁在柜子里,等你办完手续再放回去。在MySQL里,我们通常借助SELECT ... FOR UPDATE语句,再配合事务来完成。

首先,需要开启一个数据库事务。使用PDO的话,就是调用beginTransaction()方法。

接着,执行带锁的查询。比如SELECT * FROM users WHERE id = 1 FOR UPDATE。这条语句一执行,id为1的那条记录就被锁定了,直到你的事务提交或回滚,其他想修改它的事务都得等着。

立即学习“PHP免费学习笔记(深入)”;

然后,在锁的保护下,安心执行业务逻辑。例如,计算并更新用户余额:UPDATE users SET balance = balance - 100 WHERE id = 1

最后,根据业务结果,选择提交事务(commit())来确认更改并释放锁,或者回滚事务(rollback())来撤销所有操作并解锁。

二、基于版本号的乐观锁实现

与悲观锁相反,乐观锁认为冲突不常发生。它不会一开始就加锁,而是在最后提交更新时,检查一下数据在此期间有没有被别人动过。怎么检查呢?一个经典的做法是给数据表加一个version字段。

第一步,读取数据时,把当前的版本号也一并查出来:SELECT id, name, balance, version FROM users WHERE id = 1

第二步,在内存里完成业务计算,比如判断余额是否足够,并算出新的余额。

第三步,也是最关键的一步,执行条件更新:UPDATE users SET balance = ?, version = version + 1 WHERE id = ? AND version = ?。这里把之前读到的版本号作为条件,只有版本号没变,更新才会成功。

第四步,检查影响行数。如果rowCount()返回1,恭喜,更新成功。如果返回0,那就意味着在你读取之后、更新之前,数据已经被其他请求修改了,本次更新失败,通常需要重试或向用户抛出异常

三、基于时间戳的乐观锁实现

原理和版本号机制一模一样,只是校验的“信物”换成了数据行最后更新的时间戳(比如updated_at字段)。这对于那些已经存在时间戳字段、不方便再加version字段的表来说,是个很实用的选择。

首先,读取数据和当前的时间戳:SELECT id, name, balance, updated_at FROM users WHERE id = 1

接着,处理业务逻辑。

然后,执行带时间戳校验的更新:UPDATE users SET balance = ?, updated_at = NOW() WHERE id = ? AND updated_at = ?

最后,验证结果。同样,如果影响行数为0,就表明数据已被他人抢先更新,当前操作应当被拒绝

四、使用Redis实现分布式悲观锁

当你的PHP应用部署在多台服务器上时,数据库的行锁就管不住跨进程的并发了。这时候,需要一个全局都能看见的“信号灯”,Redis分布式锁就派上了用场。它的核心是利用Redis的SETNX(或带参数的SET)命令的原子性。

第一步,生成一个唯一的锁标识键,例如“lock:user:1”,其中的1可以是对应的用户ID。

第二步,尝试获取锁。执行Redis::set($key, $unique_value, ['nx', 'ex' => 10])。这条命令的意思是,只有当键不存在(nx)时才设置它,并给它一个10秒的过期时间(ex),这是为了防止客户端崩溃导致锁永远无法释放。

第三步,判断返回值。如果返回TRUE,说明成功拿到了锁,可以放心地去操作数据库了。如果返回FALSE说明锁已被他人持有,请求需要等待或直接返回“操作冲突”的提示

第四步,操作完成后,务必释放锁。注意,最好使用Lua脚本,通过比对锁值来原子性地删除键,避免误删了其他请求后来设置的锁。

五、CAS风格的乐观锁封装类

在业务代码里到处写版本号校验和重试逻辑,显然不够优雅。更好的做法是,把这些通用逻辑封装成一个工具类,这就是CAS(Compare-And-Swap)风格的工具类。它把“读取-计算-更新”这个循环流程包装起来,自动处理失败重试。

首先,定义重试次数的上限,比如默认3次,防止在极端高并发下陷入无限循环。

然后,在一个循环内,执行标准的“读取数据 -> 业务计算 -> 尝试更新”流程。每次更新失败,都重新读取最新的数据和版本号。

在每次更新前,工具类会自动校验当前内存中的版本号是否与数据库中的一致。

如果循环达到最大重试次数后更新仍未成功,工具类便会终止操作,抛出一个清晰的OptimisticLockException异常,让上层业务逻辑能妥善处理。

说到底,选择乐观锁还是悲观锁,没有绝对的优劣,关键看你的业务场景。冲突频繁、临界区操作耗时长的,悲观锁更省心;冲突概率低、追求更高吞吐量的,乐观锁往往表现更佳。理解它们的原理,并在PHP中熟练运用,是构建稳健并发系统的基本功。

来源:https://www.php.cn/faq/2409949.html

游乐网为非赢利性网站,所展示的游戏/软件/文章内容均来自于互联网或第三方用户上传分享,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系youleyoucom@outlook.com。

同类文章
更多
Composer如何查看可升级的包_Composer查看可升级包步骤

Composer如何查看可升级的包_Composer查看可升级包步骤

Composer如何查看可升级的包?别被默认输出“骗”了 直接运行 composer outdated,这大概是所有PHP开发者检查依赖更新的第一反应。但这里有个常见的误解:这个命令的输出结果,并不是在告诉你“世界上所有可用的新版本”,它只显示那些符合你composer json里既定版本约束的更新

时间:2026-05-02 22:44
Ubuntu Golang编译失败常见原因有哪些

Ubuntu Golang编译失败常见原因有哪些

Ubuntu 上 Golang 编译失败的常见原因与排查要点 在 Ubuntu 上折腾 Go 项目,编译失败这事儿,说大不大,说小不小。它不像运行时错误那样有清晰的逻辑线索,往往一个看似不起眼的配置问题,就能让整个构建过程戛然而止。别慌,咱们今天就把那些最常见的“拦路虎”梳理一遍,并提供一套清晰的排

时间:2026-05-02 22:44
PhpStorm一键导入VSCode主题(无缝切换)

PhpStorm一键导入VSCode主题(无缝切换)

PhpStorm 无法直接使用 VSCode 主题,因二者格式(JSON vs icls)、语义体系、作用域命名完全不兼容;所谓“一键导入”无官方支持且不可靠,需手动迁移核心颜色、图标与字体以实现视觉一致性。 PhpStorm 里根本不能直接用 VSCode 主题 事情是这样的:VSCode 的主

时间:2026-05-02 22:43
phpstorm怎么快速将选中代码包裹在Try-Catch中(快捷键)

phpstorm怎么快速将选中代码包裹在Try-Catch中(快捷键)

PhpStorm 中 Ctrl+Alt+T(macOS 为 Cmd+Alt+T)可快速用 try-catch 包裹代码,但需选中有效 PHP 语句且文件类型为 PHP;默认捕获 Exception,PHP 7+ 应改用 Throwable;可自定义 Live Templates 添加日志或 re

时间:2026-05-02 22:43
Ubuntu下Golang编译项目结构怎么设计

Ubuntu下Golang编译项目结构怎么设计

在Ubuntu下使用Golang编译项目时,可以遵循以下项目结构设计原则 好的项目结构是高效开发和团队协作的基石。在Ubuntu环境下用Go语言开发,遵循一些清晰的设计原则,能让编译、测试和维护都变得事半功倍。下面这套结构方案,可以说是经过大量项目验证的“最佳实践”了。 1 项目根目录 首先,为你

时间:2026-05-02 22:43
热门专题
更多
刀塔传奇破解版无限钻石下载大全 刀塔传奇破解版无限钻石下载大全
洛克王国正式正版手游下载安装大全 洛克王国正式正版手游下载安装大全
思美人手游下载专区 思美人手游下载专区
好玩的阿拉德之怒游戏下载合集 好玩的阿拉德之怒游戏下载合集
不思议迷宫手游下载合集 不思议迷宫手游下载合集
百宝袋汉化组游戏最新合集 百宝袋汉化组游戏最新合集
jsk游戏合集30款游戏大全 jsk游戏合集30款游戏大全
宾果消消消原版下载大全 宾果消消消原版下载大全
  • 日榜
  • 周榜
  • 月榜
热门教程
更多
  • 游戏攻略
  • 安卓教程
  • 苹果教程
  • 电脑教程