php事务执行顺序_PHP微服务之分布式事务

分布式事务一直是微服务的一个难点。相关的解决方案和框架大部分是 java 的,那么 php 该如何解决呢?下面一步一步讲解如何用 php 解决分布式事务。

单机单数据源事务

首先从单机事务开始。

大概逻辑如下 :

try {  // 开始事务 $db->beginTransaction(); // 执行你的操作  // ... // 提交事务 $db->commit();} catch (Exception $e) { // 执行失败 回滚 $db->rollBack();}

单机多个数据源事务

如果你业务涉及到多个数据库,事务大概逻辑是这个样子:

try {  // 开始事务 $db1->beginTransaction(); $db2->beginTransaction();  // 执行你的操作  // ... // 提交事务 $db1->commit(); $db2->commit();} catch (Exception $e) { // 执行失败 回滚 $db1->rollBack(); $db2->rollBack();}

多机多数据源事务(分布式事务)

如果你的数据源和业务代码都是分开的(微服务)这就是我们今天的核心。

由前面两种情况来看,大概逻辑是差不多的,主要也分为 4 个步骤。

  • 开始事务
  • 执行逻辑代码
  • 提交事务
  • 回滚事务

有些文章也称为 tcc 也就是 234 步骤。

我们用一个常用的例子:下单。

主要 3 个步骤:

  • 创建订单
  • 修改库存
  • 修改用户积分

假设订单,库存,用户都是独立的服务。

按照前面的经验大概分为 4 个步骤,我们以用户为例代码如下:

992880cedd9f4bcea41530ec852aefe0.png

库存 (stock),订单 (order) 和上面类似,也需要这 4 个方法,我就不写了。

难点在于我们没法直接操作数据源,只能通过 rpc 调用相应的服务来操作。依次执行上面的方法就好了。代码如下:

try {  // 开始事务 $user = new User(); $stock = new Stock(); $order = new Order(); $user = $user->beginTransaction(); $stock = $stock->beginTransaction(); $order = $order->beginTransaction(); // 执行你的操作  $user = $user->doTransaction(); $stock = $stock->doTransaction(); $order = $order->doTransaction(); // 提交事务 $user->commit(); $stock->commit(); $order->commit();} catch (Exception $e) { // 执行失败 回滚 $user->rollBack(); $stock->rollBack(); $order->rollBack();}

到这里可能有人看出问题来了,正常情况下这样肯定是不行的。要上面这段代码成立需要满足 1 个条件:User 分别调用了 3 次,也就是 3 个请求。要保证这 3 个请求是调用的同一个实例化后的对象。Stock 和 Order 一样。

User 调用逻辑如下:

// 第一次请求调用$user = new User();$user = $user->beginTransaction();// 第二次请求调用 复用的第一次 $user$user = $user->doTransaction();// 第三次请求调用 复用的第一次 $user$user->commit();//或者 $user->rollBack();

one 框架 https://github.com/lizhichao/one

极简。高性能。松耦合。分布式。可运行于多种环境

one 框架完美支持上面的要求。只需要把上面的 User、Stock 和 Order 添加为 rpc 服务即可。还需要注意 beginTransaction 和 doTransaction 方法必须返回 $this 提供给后面的方法调用。

user 服务如下:

RpcServer::add(User::class);

其他两个类似。到此分布式事务问题就搞定了,可能觉得这么简单吗?这主要由于 one框架的 rpc 服务提供了链式调用(多个请求复用同一个对象)的功能。

可能有人要问:如果因为网络问题或者其他问题导致最后一个服务的最后一次调用失败了怎么办?

解决方案就是事务补偿,你可以把这类极端的情况下的错误,放到一个队列里 起一服务来专门处理这里问题。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值