Laravel中用Redis来做任务队列

本文介绍了在Laravel中使用Redis作为任务队列时遇到的任务重复执行问题。问题源于Laravel的reserved任务超时机制,导致任务在60秒未完成时被重新投入队列。解决方案是修改代码,确保任务在开始执行前先从队列中删除,从而防止任务多次执行。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

利用Redis可以很方便的实现一个任务队列,但是在Laravel中,Redis的队列总会出现一个任务多次执行的问题。究其原因是它写死了reserved的时长,也就是如果1分钟后任务没有执行完成,那么这个任务就会被重新放回队列。下面是队列的简单使用和执行原理。

设置

设置队列使用Redis非常容易,在app/config/queue.php中配置

 
  1. ...

  2. 'default' => 'redis',

  3. ...

  4. 'connections' => array(

  5. ...

  6. 'redis' => array(

  7. 'driver' => 'redis',

  8. 'queue' => 'waa',

  9. ),

  10. ),

即可。

使用

使用时不需要多配置,只要写好Queue类和其fire方法,在需要的位置出队即可。具体方法可以看这里

 
  1. class SendEmail {

  2.  
  3. public function fire($job, $data)

  4. {

  5. //

  6. $job->delete();

  7. }

  8.  
  9. }

  10.  
  11. Queue::push('SendEmail@send', array('message' => $message));

流程

Laravel利用artisan命令来执行出队操作,然后进行任务的执行。方法调用如下:

  1. artisan queue:work
  2. WorkerCommand:fire()
  3. Worker:pop()
  4. Worker:getNextJob()
  5. RedisQueue:pop()
  6. Worker:process()

我遇到的问题就在这里,在RedisQueue:pop()方法中,有这样一句:

$this->redis->zadd($queue.':reserved', $this->getTime() + 60, $job);

这里将当前执行的任务放到另外一个reserved队列中,超时时间是60s。也就是说,如果60s后这个任务没有被删除掉,则任务会重新被放入队列中来。因此,在实际的使用过程中,任务很可能被多次执行。解决的办法是

 
  1. class SendEmail {

  2.  
  3. public function fire($job, $data)

  4. {

  5. $job->delete();

  6. // job

  7. }

  8.  
  9. }

即先删除这个任务,再开始执行任务。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值