文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

Laravel中用Observer事件致Redis队列异常问题怎么解决

2023-06-21 22:35

关注

本篇内容主要讲解“Laravel中用Observer事件致Redis队列异常问题怎么解决”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Laravel中用Observer事件致Redis队列异常问题怎么解决”吧!                      

1、业务逻辑

新建某个模型之后,利用 Observer 模型事件 Created 推入异步短信发送队列

App\Http\Controllers\UsersController

    public function store(User $user)    {        \DB::beginTransaction();        try{            $input = request()->validated();            $user->fill($input);            $user->save();            //do something......            //其他数据表操作            \DB::commit();        } catch ($e \Exception) {            \DB::rollBack();        }    }

App\Observers\UserObserver

class UserObserver{    public function created (User $user)    {        dispatch(new SmsQueue($user));    }}

2、发现异常

业务部门反馈偶尔有用户收取不到短信通知,我便查看日志发现偶尔有错误异常:No query results for model [App\Models\User]. 表示找不到对应的模型

我敲不应该啊,我是在创建模型之后再进行队列调用……,遂对业务代码再进行仔细核查猜测应该是受事务影响。

验证猜想:

    public function store(User $user)    {        \DB::beginTransaction();        try{            $input = request()->validated();            $user->fill($input);            $user->save();            //do something......            //其他数据表操作            sleep(3); //三秒之后再提交事务                        \DB::commit();        } catch ($e \Exception) {            \DB::rollBack();        }    }

果然在等待三秒之后提交队列异常已是 100%  触发。

3、原因分析

然后我在搜索 Github Issues 记录时,发现此问题在 2015 年的一个 Issue 已经有人提出,而在 Laravel 8.X 中终于新增了对事务模型事件的支持;learnku.com/docs/laravel/8.x/eloqu... ,在社区文档似乎并没有找到相关说明~

由于我的版本是 6.x 所以用不了这个新特性[哭唧唧]~~

4、解决异常

1. 修改 MySQL 事务隔离级别(不推荐)

这里涉及到 MySQL 的事务隔离级别,InnoDB 引擎的默认隔离级别是 REPEATABLE READ,关于各个级别的区别可以在 官方文档 找到。

将隔离级别切换到 READ UNCOMMITTED 即可解决此问题,但是为了防止出现更大的问题我劝你别用这种方式~

2. 增加事件监听

查看源码得知在事务完成之后,会调用对应的事件,所以只需增加对事件的监听即可。

Laravel中用Observer事件致Redis队列异常问题怎么解决

  1. 新增类 App\Handlers\TransactionHandler

    class TransactionHandler{    public array $handlers;    public function __construct()    {        $this->handlers = [];    }    public function add(\Closure $handler)    {        $this->handlers[] = $handler;    }    public function run()    {        foreach ($this->handlers as $handler) {            $handler();        }        $this->handlers = [];    }}
  2. 创建辅助函数  app/helpers.php

    if (! function_exists('after_transaction')) {        function after_transaction(Closure $job)    {        app()->singletonIf(\App\Handlers\TransactionHandler::class, function (){            return new \App\Handlers\TransactionHandler();        });        app(\App\Handlers\TransactionHandler::class)->add($job);    }}
  3. 创建监听  App\Listeners\TransactionListener

    namespace App\Listeners;use App\Handlers\TransactionHandler;class TransactionListener{    public function handle()    {        app(TransactionHandler::class)->run();    }}
  4. 绑定监听 App\Providers\EventServiceProvider

    namespace App\Providers;use App\Listeners\TransactionListener;use Illuminate\Database\Events\TransactionCommitted;use Illuminate\Database\Events\TransactionRolledBack;use Illuminate\Foundation\Support\Providers\EventServiceProvider as ServiceProvider;;class EventServiceProvider extends ServiceProvider{        protected $listen = [        TransactionCommitted::class => [            TransactionListener::class        ],        TransactionRolledBack::class => [            TransactionListener::class        ]    ];}
  5. 更改调用方式  App\Observers\UserObserver

class UserObserver{    public function created (User $user)    {        after_transaction(function() use ($user) {            dispatch(new SmsQueue($user));        });    }}

到此,相信大家对“Laravel中用Observer事件致Redis队列异常问题怎么解决”有了更深的了解,不妨来实际操作一番吧!这里是编程网网站,更多相关内容可以进入相关频道进行查询,关注我们,继续学习!

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯