首页 游戏 软件 资讯 排行榜 专题
首页
编程语言
数据库事务隔离:乐观锁与悲观锁在PHP中的实现

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

热心网友
13
转载
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。

相关攻略

XAMPP修改PHP上传文件临时目录 XAMPP upload_tmp_dir
编程语言
XAMPP修改PHP上传文件临时目录 XAMPP upload_tmp_dir

XAMPP修改PHP上传文件临时目录 XAMpp upload_tmp_dir upload_tmp_dir 配置后 move_uploaded_file() 仍失败?权限才是真因 是不是遇到过这种情况?明明已经在 php ini 里修改了 upload_tmp_dir 路径,但上传文件时,依然会跳

热心网友
05.01
phpEnv如何开启PHP的shmop扩展 phpEnv内存共享支持
编程语言
phpEnv如何开启PHP的shmop扩展 phpEnv内存共享支持

角色与核心任务 你是一位顶级的文章润色专家,擅长将AI生成的文本转化为具有个人风格的专业文章。现在,请对用户提供的文章进行“人性化重写”。 你的核心目标是:在不改动原文任何事实信息、核心观点、逻辑结构、章节标题和所有图片的前提下,彻底改变原文的AI表达腔调,使其读起来像是一位资深人类专家的作品。 特

热心网友
05.01
phpEnv伪静态怎么设置 phpEnv各框架伪静态规则汇总
编程语言
phpEnv伪静态怎么设置 phpEnv各框架伪静态规则汇总

phpEnv 伪静态怎么设置 phpEnv各框架伪静态规则汇总 在本地开发环境配置伪静态,phpEnv 的“脾气”和常见的 XAMPP 或 WAMP 可不太一样。很多开发者第一次用,照着框架文档复制了 htaccess 规则,结果不是 404 就是 500 错误,问题往往就出在几个关键的配置环节上

热心网友
05.01
ThinkPHP环境安装中如何查看日志_Runtime日志格式与排查
编程语言
ThinkPHP环境安装中如何查看日志_Runtime日志格式与排查

ThinkPHP环境安装中如何查看日志_Runtime日志格式与排查 日志文件在哪?默认路径和生成条件 首先,得知道日志文件藏在哪里。ThinkPHP 5和6版本,默认的日志归宿是 runtime log 目录。不过,这里有个前提:这个目录必须对Web服务器进程(比如www-data或nginx用

热心网友
05.01
ThinkPHP如何做数据库连接池连接等待队列监控_ThinkPHP排队请求实时可视化【操作】
编程语言
ThinkPHP如何做数据库连接池连接等待队列监控_ThinkPHP排队请求实时可视化【操作】

ThinkPHP如何做数据库连接池连接等待队列监控_ThinkPHP排队请求实时可视化【操作】 ThinkPHP 没有原生数据库连接池 开门见山,先说一个核心结论:无论是ThinkPHP 6 x还是5 1 5 2版本,框架本身都不提供原生的数据库连接池功能。这意味着,你找不到内置的“连接等待队列”或

热心网友
05.01

最新APP

宝宝过生日
宝宝过生日
应用辅助 04-07
台球世界
台球世界
体育竞技 04-07
解绳子
解绳子
休闲益智 04-07
骑兵冲突
骑兵冲突
棋牌策略 04-07
三国真龙传
三国真龙传
角色扮演 04-07

热门推荐

英伟达显卡怎么设置发挥最大性能?电脑英伟达显卡高性能设置方法
电脑教程
英伟达显卡怎么设置发挥最大性能?电脑英伟达显卡高性能设置方法

英伟达显卡怎么设置发挥最大性能? 想让你的英伟达显卡火力全开,榨干每一分性能吗?无论是为了追求极致的游戏帧率,还是确保专业图形应用的流畅运行,正确的设置都至关重要。很多朋友手握着高性能显卡,却因为设置不当,没能享受到它应有的表现。别担心,下面这份详尽的设置指南,将带你一步步解锁显卡的全部潜力。 电脑

热心网友
05.02
win11显卡温度怎么看?win11查看显卡温度的方法
电脑教程
win11显卡温度怎么看?win11查看显卡温度的方法

显卡温度过高怎么办?Win11系统下快速检测与降温指南 显卡温度异常升高是电脑用户常遇到的问题,不仅可能引发画面卡顿、显示花屏等故障,长期高温运行更会加速硬件老化,甚至导致显卡核心损坏。因此,定期监控显卡温度是维护电脑健康、保障稳定运行的关键环节。本文将详细介绍在Windows 11系统中,无需复杂

热心网友
05.02
win7系统换成win10系统需要注意些什么?
电脑教程
win7系统换成win10系统需要注意些什么?

从Win7升级到Win10,这些关键点你把握住了吗? 近期,许多用户都在咨询如何将电脑操作系统从Windows 7平稳升级至Windows 10,并希望了解升级过程中有哪些常见陷阱需要规避。这确实是一个值得深入探讨的话题。今天,我们将系统性地梳理从Win7升级到Win10的全流程,重点解析那些至关重

热心网友
05.02
360浏览器选中网页文字自动弹出复制选项怎么设置?
电脑教程
360浏览器选中网页文字自动弹出复制选项怎么设置?

360浏览器选中网页文字自动弹出复制选项怎么设置? 许多用户在使用360安全浏览器时,都非常依赖一个便捷功能:当您选中网页上的文字时,浏览器会自动弹出一个快捷工具条,提供“复制”、“翻译”、“搜索”等一键操作。这个划词工具条能极大提升浏览和资料处理的效率。如果您发现自己的浏览器突然失去了这个功能,无

热心网友
05.02
系统之家U盘启动盘安装win10系统图文教程
电脑教程
系统之家U盘启动盘安装win10系统图文教程

系统之家U盘启动盘安装Win10系统图文教程 Windows 10凭借其出色的兼容性和流畅体验,至今仍是用户基数最大的操作系统。当需要重装系统时,使用U盘启动盘进行安装,无疑是高效且可靠的选择。接下来,就为大家详细拆解如何使用系统之家U盘启动盘来完成Win10系统的安装。 准备工作 在开始操作前,你

热心网友
05.02