侯体宗的博客
  • 首页
  • Hyperf版
  • beego仿版
  • 人生(杂谈)
  • 技术
  • 关于我
  • 更多分类
    • 文件下载
    • 文字修仙
    • 中国象棋ai
    • 群聊
    • 九宫格抽奖
    • 拼图
    • 消消乐
    • 相册

Laravel5.5+基于reset机制实现分布式事务,了解一下laravel-reset-transaction

Laravel  /  管理员 发布于 3年前   1166

之前想了解一下这方面功能包,在git上看到了laravel-reset-transaction包,感觉很好,

所以记录一下;后面有项目在实践一下


准备软件依赖包

laravel:5.5 - 8.0之间
composer:2.0

安装composer包 - laravel-reset-transaction

#必须使用composer2版本
composer require windawake/laravel-reset-transaction dev-master


首先创建 控制器/模型/数据库表

ResetProductController.php 控制器,
创建 ResetProductModel.php 模型,
创建 reset_transaction 和 reset_product 两张数据库表。

这些操作只需要执行下面命令全部完成

php artisan resetTransact:create-examples  


phpunit.xml 增加 testsuite Transaction

<?xml version="1.0" encoding="UTF-8"?>
<phpunit xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:noNamespaceSchemaLocation="./vendor/phpunit/phpunit/phpunit.xsd"
         bootstrap="vendor/autoload.php"
         colors="true">
    <testsuites>
        ......
        <testsuite name="Transaction">
            <directory>./vendor/windawake/laravel-reset-transaction/tests</directory>
        </testsuite>
    </testsuites>
    ......
</phpunit>


然后启动web服务器

php artisan serve --host=0.0.0.0 --port=8000


最后运行测试命令 

./vendor/bin/phpunit --testsuite=Transaction

运行结果如下所示,5 个例子测试通过。

oot@DESKTOP-VQOELJ5:/web/linux/php/laravel/laravel62# ./vendor/bin/phpunit --testsuite=Transaction
PHPUnit 8.5.20 by Sebastian Bergmann and contributors.
.....                                                               5 / 5 (100%)
Time: 219 ms, Memory: 22.00 MB
OK (5 tests, 5 assertions)


功能特性:

开箱即用,不需要重构原有项目的代码,与 mysql 事务写法一致,简单易用。

遵守两段提交协议,属于强一致性事务,高并发下,支持读已提交的事务隔离级别。

由于事务拆分成多个,变成了几个小事务,压测发现比 mysql 普通事务更少发生死锁。

支持事务嵌套,与 savepoint 一致效果。

支持避免不同业务代码并发造成脏数据的问题。

默认支持 http 协议的服务化接口,想要支持其它协议则需要重写中间件。


解决了哪些并发场景:

1.一个待发货订单,用户同时操作发货和取消订单,只有一个成功
2.积分换取优惠券,只要出现积分不够扣减或者优惠券的库存不够扣减,就会全部失败。

原理解析

Reset Transaction,中文名为重置型分布式事务,又命名为 RT 模式,

与 seata AT 模式都是属于二段提交。

看过《明日边缘》电影就会知道,存档和读档的操作。

这个分布式事务组件仿造《明日边缘》电影的原理,


reset 是重置的意思,即每次请求基础服务一开始时读档,

然后继续后面的操作,

结束时所有操作全部回滚并且存档,

最后一步 commit 把存档全部执行成功。


整个过程是遵守两段提交协议,先 prepare,最后 commit。

以用户 A 用招行卡转账 100 元给用户 B 招行账号的场景为例子,画了以下流程图。

Reset Transaction.png


右图开启 reset 分布式事务后,

比左图多了请求 4。

请求 4 所做的事情,都是请求 1-3 之前做过的东西,又回来原点重新再来,

最终提交事务,结束这转账的流程。


如何使用

以 vendor/windawake/laravel-reset-transaction/tests/TransactionTest.php 文件为例子
<?php
namespace Tests\Feature;
use Tests\TestCase;
use Illuminate\Support\Facades\DB;
class TransactionTest extends TestCase
{
    public function testCreateWithCommit()
    {
        $num = rand(1, 10000);
        $productName = 'php ' . $num;
        $data = [
            'store_id' => 1,
            'product_name' => $productName,
        ];
        // 开启分布式事务,其实是生成全局唯一id
        $transactId = $this->beginDistributedTransaction();
        $header = [
           在header 'transact_id' => $transactId,
        ];
        // 分布式事务内,请求都需要在request header带上transact_id
        $response = $this->post('api/resetProduct', $data, $header);
        $product = $response->json();
        // 分布式事务提交,也是接口请求,把之前的存档记录全部处理
        $this->commitDistributedTransaction($transactId);
        $response = $this->get('/api/resetProduct/' . $product['pid']);
        $product = $response->json();
        $this->assertEquals($productName, $product['product_name']);
    }
    private function beginDistributedTransaction()
    {
        return session_create_id();
    }
    private function commitDistributedTransaction($transactId)
    {
        $response = $this->post('/api/resetTransaction/commit', [], ['transact_id' => $transactId]);
        return $response->getStatusCode();
    }
    private function rollbackDistributedTransaction($transactId)
    {
        $response = $this->post('/api/resetTransaction/rollback', [], ['transact_id' => $transactId]);
        return $response->getStatusCode();
    }
}


git地址:实现laravel、hyperf

https://github.com/windawake/laravel-reset-transaction
https://github.com/windawake/hyperf-reset-transaction


  • 上一条:
    千万别自建翻墙vpn,VPN翻墙被抓行政处罚案例越来越多
    下一条:
    go+beego框架中使用无极限原理数据处理实现详情页评论展示功能
  • 昵称:

    邮箱:

    0条评论 (评论内容有缓存机制,请悉知!)
    最新最热
    • 分类目录
    • 人生(杂谈)
    • 技术
    • linux
    • Java
    • php
    • 框架(架构)
    • 前端
    • ThinkPHP
    • 数据库
    • 微信(小程序)
    • Laravel
    • Redis
    • Docker
    • Go
    • swoole
    • Windows
    • Python
    • 苹果(mac/ios)
    • 相关文章
    • Laravel 11.15版本发布 - Eloquent Builder中添加的泛型(0个评论)
    • Laravel 11.14版本发布 - 新的字符串助手和ServeCommand改进(0个评论)
    • Laravel 11.12版本发布 - Artisan的`make`命令自动剪切`.php `扩展(0个评论)
    • Laravel的轻量型购物车扩展包:binafy/laravel-cart(0个评论)
    • Laravel 11.11版本发布 - 查看模型中的第三方关系:show(0个评论)
    • 近期文章
    • 在go中实现一个常用的先进先出的缓存淘汰算法示例代码(0个评论)
    • 在go+gin中使用"github.com/skip2/go-qrcode"实现url转二维码功能(0个评论)
    • 在go语言中使用api.geonames.org接口实现根据国际邮政编码获取地址信息功能(1个评论)
    • 在go语言中使用github.com/signintech/gopdf实现生成pdf分页文件功能(0个评论)
    • gmail发邮件报错:534 5.7.9 Application-specific password required...解决方案(0个评论)
    • 欧盟关于强迫劳动的规定的官方举报渠道及官方举报网站(0个评论)
    • 在go语言中使用github.com/signintech/gopdf实现生成pdf文件功能(0个评论)
    • Laravel从Accel获得5700万美元A轮融资(0个评论)
    • 在go + gin中gorm实现指定搜索/区间搜索分页列表功能接口实例(0个评论)
    • 在go语言中实现IP/CIDR的ip和netmask互转及IP段形式互转及ip是否存在IP/CIDR(0个评论)
    • 近期评论
    • 122 在

      学历:一种延缓就业设计,生活需求下的权衡之选中评论 工作几年后,报名考研了,到现在还没认真学习备考,迷茫中。作为一名北漂互联网打工人..
    • 123 在

      Clash for Windows作者删库跑路了,github已404中评论 按理说只要你在国内,所有的流量进出都在监控范围内,不管你怎么隐藏也没用,想搞你分..
    • 原梓番博客 在

      在Laravel框架中使用模型Model分表最简单的方法中评论 好久好久都没看友情链接申请了,今天刚看,已经添加。..
    • 博主 在

      佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 @1111老铁这个不行了,可以看看近期评论的其他文章..
    • 1111 在

      佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 网站不能打开,博主百忙中能否发个APP下载链接,佛跳墙或极光..
    • 2016-10
    • 2016-11
    • 2017-07
    • 2017-08
    • 2020-03
    • 2020-04
    • 2020-05
    • 2020-06
    • 2020-07
    • 2020-08
    • 2020-09
    • 2020-10
    • 2020-11
    • 2021-01
    • 2021-02
    • 2021-03
    • 2021-04
    • 2021-05
    • 2021-06
    • 2021-07
    • 2021-08
    • 2021-09
    • 2021-10
    • 2021-11
    • 2021-12
    • 2022-01
    • 2022-02
    • 2022-03
    • 2022-04
    • 2022-05
    • 2022-06
    • 2022-07
    • 2022-08
    • 2022-09
    • 2022-10
    • 2022-11
    • 2022-12
    • 2023-01
    • 2023-02
    • 2023-03
    • 2023-04
    • 2023-05
    • 2023-06
    • 2023-07
    • 2023-08
    • 2023-09
    • 2023-10
    • 2023-11
    • 2023-12
    • 2024-01
    • 2024-02
    • 2024-03
    • 2024-04
    • 2024-05
    • 2024-06
    • 2024-07
    Top

    Copyright·© 2019 侯体宗版权所有· 粤ICP备20027696号 PHP交流群

    侯体宗的博客