Laravel 队列使用的实现


Posted in PHP onJanuary 08, 2019

1 环境

Laravel是一种类似ThinkPHP的php框架,封装的诸多功能可以很方便的使用。队列Queue便是其中之一。

Windows环境下,可使用PHPstorm作为Laravel的集成开发环境IDE。

2 队列

Laravel可配置多种队列驱动,包括 "sync", "database", "beanstalkd", "sqs", "redis", "null"(具体参见app/config/queue.php)
其中sync为同步,database为使用数据库,后面三种为第三方队列服务,最后一种为不使用队列。

通过在 .env 中的 QUEUE_CONNECTION 选项,来决定选择何种驱动。

如 QUEUE_CONNECTION=database 即为选择数据库驱动队列。

3 原理

所谓队列,会有数据的生产者和消费者之分。生产者向队列中投递数据,消费者从队列中获取数据。

比如向用户发送邮件的场景:现在有10w封邮件需要发送,最简单的,我们需要有一个方法将邮件的收件人、内容等,拆分成10w条任务放在队列中,同时需要设置一个回调方法负责处理每条任务。当队列中有邮件发送任务时,队列会主动调用回调方法,并传递任务详情进去。回调方法处理完成后,单条邮件即发送完毕。其他邮件依样处理。

4 使用数据库驱动队列

4.1 生成任务表

在终端下输入

php artisan queue:table
php artisan migrate

在数据库连接正常的情况下,会在数据库中出现jobs表:

[id] bigint 
 [queue] nvarchar(255) 
 [payload] nvarchar(max) 
 [attempts] tinyint 
 [reserved_at] int 
 [available_at] int 
 [created_at] int

4.2 创建任务类

php artisan make:job SendEmail

在终端内执行上述命令,会自动生成 app/Jobs/SendMail.php 文件

class SendMail implements ShouldQueue

在该文件的handle方法中,可以放置任务处理逻辑。

4.3 发送任务

在任意位置,均可像下面一样调用 dispatch 发送任务

SendMail::dispatch($email);

4.4 驱动队列

完成上述步骤后,可以在数据库中发现一条记录(导出为insert SQL语句):

INSERT INTO [jobs]([id], [queue], [payload], [attempts], [reserved_at], [available_at], [created_at]) VALUES (6, N'default', N'{"displayName":"App\\Jobs\\ProcessPodcast","job":"Illuminate\\Queue\\CallQueuedHandler@call","maxTries":null,"timeout":null,"timeoutAt":null,"data":{"commandName":"App\\Jobs\\ProcessPodcast","command":"O:23:\"App\\Jobs\\ProcessPodcast\":8:{s:29:\"\u0000App\\Jobs\\ProcessPodcast\u0000data\";s:6:\"111222\";s:6:\"\u0000*\u0000job\";N;s:10:\"connection\";N;s:5:\"queue\";N;s:15:\"chainConnection\";N;s:10:\"chainQueue\";N;s:5:\"delay\";N;s:7:\"chained\";a:0:{}}"}}', 0, NULL, 1545980176, 1545980176);

此时任务已经放置在数据库内,只有将队列运行起来后,队列才能主动调用回调方法。

php artisan queue:work

在终端内运行上述命令即可。该命令还有诸多参数,如deamon、tries等,可根据需要指定。

4.5 守护进程

为了保证应用服务的稳定性,需要开启守护进程。

Linux下,一般使用 Supervisor ,Windows下使用 Forever

4.6 执行失败的处理

对于处理失败的任务,Laravel也提供的解决方案。通过运行如下命令,即可创建表以记录失败任务。

php artisan queue:failed-table
php artisan migrate

在数据库中即生成 failed_jobs :

[id] bigint
 [connection] nvarchar(max)
 [queue] nvarchar(max) 
 [payload] nvarchar(max) 
 [exception] nvarchar(max) 
 [failed_at] datetime

导致任务失败的 Exception 会被传递到 SendMail 的 failed 方法,因而你需要在SendMail中自行实现该方法,并做进一步处理。

任务执行失败的原因有很多,如传参错误、尝试次数超过限制、超时、甚至在 handle 方法中抛出异常,均会作为失败任务处理。

4.7 任务执行前后的处理

Laravel提供了任务执行前后的处理入口,即在 App/Providers/AppServiceProvider 中的 boot() 中加入如下代码:

public function boot()
{
   Queue::before( function (JobProcessing $event) {
     Log::info("处理任务前");
   });
   Queue::after( function (JobProcessed $event) {
     Log::info("处理任务后");
   });
}

传递的 $event 中,带有任务详情,几个简单的例子:

$event->connectionName
$event->job
$event->job->payload()

5 使用 Redis 驱动队列

5.1 Laravel 安装 Predis 包

在 Laravel 中使用 Redis 之前,需要通过 Composer 安装 predis/predis 包:

composer require predis/predis

上述拓展是帮助Laravel与Redis打交道的,我们现在还缺少Redis服务。

如果此时将 .env 中的 QUEUE_CONNECTION 改为 redis,访问时会报错:

Predis \ Connection \ ConnectionException (10061)
����Ŀ����������ܾ����޷����ӡ� [tcp://127.0.0.1:6379]

5.2 配置 Redis 服务

在Redis官网 下载源码后自行编译即可。

官方并未提供Windows版,Redis的Windows版式由微软工作组维护的,你可以从其GitHub页 找到。不过貌似已经不再维护了,最新的版本是16年发布的3.2.100。

Linux下通过简单的运行

./redis-server

即可开启服务,再通过

./redis-cli

来尝试使用Redis。使用也很简单,就是 set key value 和 get key。
Windows下安装后,在命令行中 cd 到安装目录

C:\Program Files\Redis>redis-server redis.windows.conf
C:\Program Files\Redis>netstat -an|find "6379"
 TCP  127.0.0.1:6379     0.0.0.0:0       LISTENING

即可开启Redis服务。

Redis服务的停止是通过如下命令:

C:\Program Files\Redis>redis-server --service-stop

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

PHP 相关文章推荐
使用ThinkPHP自带的Http类下载远程图片到本地的实现代码
Aug 02 PHP
ThinkPHP写第一个模块应用
Feb 20 PHP
PHP字符串的编码问题的详细介绍
Apr 27 PHP
php递归删除目录下的文件但保留的实例分享
May 10 PHP
PHP is_subclass_of函数的一个BUG和解决方法
Jun 01 PHP
ThinkPHP3.1新特性之内容解析输出详解
Jun 19 PHP
PHP随机生成信用卡卡号的方法
Mar 23 PHP
php实现通用的信用卡验证类
Mar 24 PHP
php array_reverse 以相反的顺序返回数组实例代码
Apr 11 PHP
php实现的后台表格分页功能示例
Oct 23 PHP
php封装db类连接sqlite3数据库的方法实例
Dec 19 PHP
php操作redis常见方法示例【key与value操作】
Apr 14 PHP
laravel 框架配置404等异常页面
Jan 07 #PHP
PHP array_shift()用法实例分析
Jan 07 #PHP
PHP parse_ini_file函数的应用与扩展操作示例
Jan 07 #PHP
PHP一个简单的无需刷新爬虫
Jan 05 #PHP
PHP智能识别收货地址信息实例
Jan 05 #PHP
PHP数字金额转换成中文大写显示
Jan 05 #PHP
PHP yield关键字功能与用法分析
Jan 03 #PHP
You might like
如何使用动态共享对象的模式来安装PHP
2006/10/09 PHP
整理的一些实用WordPress后台MySQL操作命令
2013/01/07 PHP
php实现json编码的方法
2015/07/30 PHP
微信支付开发发货通知实例
2016/07/12 PHP
PHP实现路由映射到指定控制器
2016/08/13 PHP
php微信公众平台开发之微信群发信息
2016/09/13 PHP
phpstudy2020搭建站点的实现示例
2020/10/30 PHP
json格式化/压缩工具 Chrome插件扩展版
2010/05/25 Javascript
javaScript矢量图表库-gRaphael几行代码实现精美的条形图/饼图/点图/曲线图
2013/01/09 Javascript
jQuery制作的别致导航有阴影背景高亮模式窗口
2014/04/15 Javascript
JavaScript中的事件委托及好处
2016/07/12 Javascript
基于百度地图实现产品销售的单位位置查看功能设计与实现
2016/10/21 Javascript
简单易懂的天气插件(代码分享)
2017/02/04 Javascript
p5.js入门教程之平滑过渡(Easing)
2018/03/16 Javascript
基于JS实现html中placeholder属性提示文字效果示例
2018/04/19 Javascript
vue登录页面cookie的使用及页面跳转代码
2019/07/10 Javascript
利用Angular7开发一个Radio组件的全过程
2019/07/11 Javascript
微信小程序后端(java)开发流程的详细步骤
2019/11/13 Javascript
token 机制和实现方式
2020/12/15 Javascript
在Python中使用NLTK库实现对词干的提取的教程
2015/04/08 Python
Python代码实现KNN算法
2017/12/20 Python
SVM基本概念及Python实现代码
2017/12/27 Python
Html5移动端弹幕动画实现示例代码
2018/08/27 HTML / CSS
整理HTML5中支持的URL编码与字符编码
2016/02/23 HTML / CSS
芬兰设计商店美国:Finnish Design Shop US
2019/03/25 全球购物
介绍一下UNIX启动过程
2013/11/14 面试题
职业教育毕业生求职信
2013/11/09 职场文书
采购助理岗位职责
2014/02/16 职场文书
公益广告语集锦
2014/03/13 职场文书
社区党务公开实施方案
2014/03/18 职场文书
《登鹳雀楼》教学反思
2014/04/09 职场文书
2014年教研工作总结
2014/12/06 职场文书
2014年团支部年度工作总结
2014/12/24 职场文书
中学教师读书笔记
2015/07/01 职场文书
护士自荐信范文(2016推荐篇)
2016/01/28 职场文书
SQL Server作业失败:无法确定所有者是否有服务器访问权限的解决方法
2021/06/30 SQL Server