• laravel 发生异常时回滚数据库变化


    您是否遇到过在laravel cms 控制器中编写一些代码将一堆数据写入数据库的情况?

    我经常构建接受大量数据负载的 API 端点,然后我们将一条记录写入数据库,然后使用关系将一堆相关数据写入这条新记录。

    问题是,要将其全部正确存储,您的多个数据库写入都需要全部成功。否则,您将得到该实体的不完整表示。

    有时我没有正确验证某些东西,或者由于某种原因在写入相关数据时出现异常。抛出一个异常,并且您的用户被告知出现了问题,但您仍然保留了一半的信息。

    我最近有另一个案例,我用一个新项目扼杀了一个遗留项目。新项目将接收请求,将其存储在新项目中,然后需要将其中一些数据写回遗留项目,以便遗留系统仍然可以运行。由于此数据需要在两个系统之间保持同步,如果对任一系统的写入失败,我需要回滚所有更改的记录。

    使用数据库事务很容易实现这一点。数据库事务允许我们进行一系列更改,使用这些更改的数据,然后在我们准备好时持久化它。

    当您在多条路线上执行此操作时,您需要一种方法来轻松地将此逻辑应用于需要它的每条路线。

    那么我们如何才能采用数据库事务的概念并将请求包装在其中呢?

    带中间件!

    我们可以编写一些中间件,将请求包装在数据库事务中,并在一切成功时提交事务。

    这看起来像这样:

    1. use Illuminate\Support\Facades\DB;
    2. class WrapRequestInDatabaseTransaction
    3. {
    4. public function handle($request, Closure $next)
    5. {
    6. DB::beginTransaction();
    7. $response = $next($request);
    8. if ($response->exception) {
    9. DB::rollBack();
    10. return $response;
    11. }
    12. DB::commit();
    13. return $response;
    14. }
    15. }
    1. 在数据库中创建事务
    2. 运行请求并将响应分配给$response- 此时请求的其余部分已运行,这就是我们要返回给用户的内容
    3. 检查是否$response是异常
    4. 如果是,则回滚事务并返回响应。无论如何,用户将看到他们从异常中看到的任何内容,但是在此期间对数据库的任何更改/写入/删除等都将回滚,因此它们永远不会持久化
    5. 如果一切顺利并且没有异常,将更改/写入/删除等提交到数据库并返回响应

    您也可以更改它以将事务也放在特定连接上。事实上,如果确实需要,我们可以包装多个数据库连接。如果我们以上面的第二个例子为例,我正在扼杀一个遗留项目,那么中间件在多个连接上的样子如下:

    1. use Illuminate\Support\Facades\DB;
    2. class WrapRequestInDatabaseTransaction
    3. {
    4. public function handle($request, Closure $next)
    5. {
    6. DB::beginTransaction();
    7. DB::connection('another_connection')->beginTransaction();
    8. $response = $next($request);
    9. if ($response->exception) {
    10. DB::rollBack();
    11. DB::connection('another_connection')->rollBack();
    12. return $response;
    13. }
    14. DB::commit();
    15. DB::connection('another_connection')->commit();
    16. return $response;
    17. }
    18. }

    这对于我需要它的用例非常有效,希望它也能帮助你 

  • 相关阅读:
    最大连续子序列(暑假每日一题 27)
    中秋假期,回不了家的程序员,竟然用Python做了这件事...
    东方通部署vue项目
    CleanMyMac最新版V4.11.4版MAC电脑系统加速器
    用微服务平台框架,实现高效的流程化办公!
    C语言中的 指针 和 引用的区别
    【SpringBoot】SpringBoot2之编写第一个HelloWorld
    数据库设计三大范式
    Vue--Router--嵌套路由(children)的用法
    数字图像处理与Python实现-Scikit-Image-图像特征(三)
  • 原文地址:https://blog.csdn.net/lemqs0123/article/details/127955486