导读:本期聚焦于小伙伴创作的《如何在PHP中处理数据库长事务导致的行锁占用并拆分大事务逻辑》,敬请观看详情,探索知识的价值。以下视频、文章将为您系统阐述其核心内容与价值。如果您觉得《如何在PHP中处理数据库长事务导致的行锁占用并拆分大事务逻辑》有用,将其分享出去将是对创作者最好的鼓励。

在PHP业务开发中,当我们需要对多张表或者大量数据进行批量操作时,很容易写出执行时间很长的事务。这类长事务会长时间持有数据库的行锁,导致其他请求无法操作被锁定的数据,最终引发业务阻塞甚至系统卡顿。

如何在PHP中处理数据库长事务导致的行锁占用并拆分大事务逻辑

长事务导致行锁占用的原因

数据库的行锁是在事务执行过程中对操作的数据加锁,只有事务提交或者回滚之后才会释放锁。如果事务执行时间过长,锁的持有时间就会同步变长,其他需要操作同一批数据的事务就会被阻塞,直到锁被释放才能继续执行。常见的长事务场景包括批量更新上万条数据、在事务中调用外部接口、在事务中执行复杂的查询逻辑等。

如何识别数据库中的长事务和行锁占用

我们可以通过数据库自带的诊断命令来查看当前运行的事务情况,以MySQL为例,可以执行以下语句查看运行时间超过60秒的事务:

-- 查看当前运行的事务,TRX_STARTED是事务开始时间,通过计算可得运行时长
SELECT trx_id, trx_state, trx_started, trx_mysql_thread_id
FROM information_schema.INNODB_TRX
WHERE TIMESTAMPDIFF(SECOND, trx_started, NOW()) > 60;

如果需要查看当前的行锁等待情况,可以执行以下语句:

-- 查看行锁等待关系
SELECT r.trx_id waiting_trx_id, r.trx_mysql_thread_id waiting_thread,
       b.trx_id blocking_trx_id, b.trx_mysql_thread_id blocking_thread
FROM information_schema.INNODB_LOCK_WAITS w
INNER JOIN information_schema.INNODB_TRX b ON b.trx_id = w.blocking_trx_id
INNER JOIN information_schema.INNODB_TRX r ON r.trx_id = w.requesting_trx_id;

拆分大事务的核心思路

拆分大事务的核心原则是减少单个事务的操作范围和执行时长,让锁尽可能快地释放,具体可以从以下几个方向入手:

  • 按业务维度拆分:如果事务中包含多个独立的业务操作,把每个业务操作拆成独立的事务,避免所有操作放在同一个事务中执行。
  • 按数据量拆分:如果是批量操作大量数据,把批量操作拆成多次小批量操作,每次操作少量数据就提交一次事务。
  • 减少事务内的非必要操作:把查询、外部接口调用、日志写入等不需要加锁的操作移到事务外部,只把必须加锁的写操作放在事务里。

PHP中拆分大事务的代码示例

场景1:批量更新用户积分

假设我们需要给10万用户增加积分,原来的长事务写法会把所有更新放在一个事务里:

<?php
// 原来的长事务写法,风险很高
$pdo = new PDO('mysql:host=127.0.0.1;dbname=test;charset=utf8', 'root', '123456');
$pdo->beginTransaction();
try {
    $userIds = range(1, 100000); // 假设要更新的用户ID列表
    $sql = 'UPDATE user SET score = score + 10 WHERE id = :id';
    $stmt = $pdo->prepare($sql);
    foreach ($userIds as $userId) {
        $stmt->execute([':id' => $userId]);
    }
    $pdo->commit();
} catch (Exception $e) {
    $pdo->rollBack();
    echo '更新失败:' . $e->getMessage();
}
?>

拆分后的写法,每次更新1000条就提交一次事务:

<?php
// 拆分后的小事务写法,每次处理1000条数据
$pdo = new PDO('mysql:host=127.0.0.1;dbname=test;charset=utf8', 'root', '123456');
$userIds = range(1, 100000);
$batchSize = 1000; // 每批处理的数据量
$total = count($userIds);

for ($i = 0; $i < $total; $i += $batchSize) {
    $batchIds = array_slice($userIds, $i, $batchSize);
    $pdo->beginTransaction();
    try {
        $sql = 'UPDATE user SET score = score + 10 WHERE id = :id';
        $stmt = $pdo->prepare($sql);
        foreach ($batchIds as $userId) {
            $stmt->execute([':id' => $userId]);
        }
        $pdo->commit();
        echo '第' . ($i / $batchSize + 1) . '批更新完成' . PHP_EOL;
    } catch (Exception $e) {
        $pdo->rollBack();
        echo '第' . ($i / $batchSize + 1) . '批更新失败:' . $e->getMessage() . PHP_EOL;
        // 可以根据需要决定是否继续处理下一批
    }
}
?>

场景2:多业务操作拆分

假设一个事务中需要完成用户余额扣减、订单创建、库存扣减三个操作,原来的写法是三个操作放在同一个事务里:

<?php
// 原来的合并事务写法
$pdo = new PDO('mysql:host=127.0.0.1;dbname=test;charset=utf8', 'root', '123456');
$pdo->beginTransaction();
try {
    // 扣减用户余额
    $pdo->exec('UPDATE user SET balance = balance - 100 WHERE id = 1');
    // 创建订单
    $pdo->exec('INSERT INTO order (user_id, amount) VALUES (1, 100)');
    // 扣减库存
    $pdo->exec('UPDATE goods SET stock = stock - 1 WHERE id = 1');
    $pdo->commit();
} catch (Exception $e) {
    $pdo->rollBack();
}
?>

拆分后每个业务操作独立成事务,减少锁占用时间:

<?php
// 拆分后的独立事务写法
$pdo = new PDO('mysql:host=127.0.0.1;dbname=test;charset=utf8', 'root', '123456');

// 1. 扣减用户余额,独立事务
$pdo->beginTransaction();
try {
    $pdo->exec('UPDATE user SET balance = balance - 100 WHERE id = 1');
    $pdo->commit();
} catch (Exception $e) {
    $pdo->rollBack();
    die('余额扣减失败');
}

// 2. 创建订单,独立事务
$pdo->beginTransaction();
try {
    $pdo->exec('INSERT INTO order (user_id, amount) VALUES (1, 100)');
    $pdo->commit();
} catch (Exception $e) {
    $pdo->rollBack();
    die('订单创建失败');
}

// 3. 扣减库存,独立事务
$pdo->beginTransaction();
try {
    $pdo->exec('UPDATE goods SET stock = stock - 1 WHERE id = 1');
    $pdo->commit();
} catch (Exception $e) {
    $pdo->rollBack();
    die('库存扣减失败');
}
?>

注意事项

拆分事务之后需要注意数据一致性的问题,如果多个事务之间有强依赖关系,拆分后可能需要增加额外的校验逻辑,比如扣减库存之前再次检查订单是否创建成功。另外批量拆分的时候,建议给每批操作增加重试机制,避免某一批操作失败导致整体任务中断。如果业务上必须保证多个操作的原子性,不能拆分事务,那么可以优化事务内的逻辑,把不必要的操作移出去,尽量缩短事务的执行时间。

PHP数据库长事务行锁占用大事务拆分修改时间:2026-06-16 08:48:39

免责声明:​ 已尽一切努力确保本网站所含信息的准确性。网站内容多为原创整理与精心编撰,观点力求客观中立。本站旨在免费分享,内容仅供个人学习、研究或参考使用。若引用了第三方作品,版权归原作者所有。如内容涉及您的权益,请联系我们处理。
内容垂直聚焦
专注技术核心技术栏目,确保每篇文章深度聚焦于实用技能。从代码技巧到架构设计,为用户提供无干扰的纯技术知识沉淀,精准满足专业提升需求。
知识结构清晰
覆盖从开发到部署的全链路。AI、前端、编程、数据库、服务器、建站、系统层层递进,构建清晰学习路径,帮助用户系统化掌握开发与运维所需的核心技术。
深度技术解析
拒绝泛泛而谈,深入技术细节与实践难点。无论是数据库优化还是服务器配置,均结合真实场景与代码示例进行剖析,致力于提供可直接应用于工作的解决方案。
专业领域覆盖
精准对应开发生命周期。从前端界面到后端编程,从数据库操作到服务器运维,形成完整闭环,一站式满足全栈工程师和运维人员的技术需求。
即学即用高效
内容强调实操性,步骤清晰、代码完整。用户可根据教程直接复现和应用于自身项目,显著缩短从学习到实践的距离,快速解决开发中的具体问题。
持续更新保障
专注既定技术方向进行长期、稳定的内容输出。确保各栏目技术文章持续更新迭代,紧跟主流技术发展趋势,为用户提供经久不衰的学习价值。