返回顶部
首页 > 资讯 > 精选 >Laravel会不会重复执行同一个队列任务
  • 728
分享到

Laravel会不会重复执行同一个队列任务

2023-06-14 22:06:44 728人浏览 薄情痞子
摘要

这篇文章主要介绍Laravel会不会重复执行同一个队列任务,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!Laravel 是什么Laravel 是一套简洁、优雅的PHP web开发框架。它可以让你从面条一样杂乱的代码中

这篇文章主要介绍Laravel会不会重复执行同一个队列任务,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!

Laravel 是什么

Laravel 是一套简洁、优雅的PHP web开发框架。它可以让你从面条一样杂乱的代码中解脱出来;它可以帮你构建一个完美的网络APP,而且每行代码都可以简洁、富于表达力。

在 Laravel 中使用 Redis 处理队列任务,框架提供的功能非常强大,但是最近遇到一个问题,就是发现一个任务被多次执行,这是为什么呢?

先说原因:因为在 Laravel 中如果一个队列(任务)执行时间大于 60 秒,就会被认为执行失败并重新加入队列中,这样就会导致重复执行同一个任务。

这个任务的逻辑就是给用户推送内容,需要根据队列内容取出用户并遍历,通过请求后端 Http 接口发送。比如有 10000 个用户,在用户数量多或接口处理速度没那么快的情况下,执行时间肯定会大于 60 秒,于是这个任务就被重新加入队列。情况更糟糕一点,前面的任务如果都没有在 60 秒执行完,就都会重新加入队列,这样同一个任务就不止重复执行一次了,而是多次。

下面从 Laravel 源代码找一下罪魁祸首。

源代码文件:vendor/laravel/framework/src/Illuminate/Queue/RedisQueue.php

protected $expire = 60;

这个 $expire 成员变量是一个固定的值,Laravel 认为一个队列再怎么 60 秒也该执行完了吧。取队列方法:

public function pop($queue = null){    $original = $queue ?: $this->default;     $queue = $this->getQueue($queue);     $this->migrateExpiredJobs($queue.':delayed', $queue);     if (! is_null($this->expire)) {        $this->migrateExpiredJobs($queue.':reserved', $queue);    }     list($job, $reserved) = $this->getConnection()->eval(        luaScripts::pop(), 2, $queue, $queue.':reserved', $this->getTime() + $this->expire    );     if ($reserved) {        return new RedisJob($this->container, $this, $job, $reserved, $original);    }}

取队列有几步操作,因为队列执行失败,或执行超时等都会放入另外的集合保存起来,以便重试,过程如下:

把因执行失败的队列从 delayed 集合重新 rpush 到当前执行的队列中。

把因执行超时的队列从 reserved 集合重新 rpush 到当前执行的队列中。

然后才是从队列中取任务开始执行,同时把队列放入 reserved 的有序集合。

这里使用了 eval 命令执行这个过程,用到了几个 lua 脚本。

从要执行的队列中取任务:

local job = redis.call('lpop', KEYS[1])local reserved = falseif(job ~= false) then    reserved = cJSON.decode(job)    reserved['attempts'] = reserved['attempts'] + 1    reserved = cjson.encode(reserved)    redis.call('zadd', KEYS[2], ARGV[1], reserved)endreturn {job, reserved}

可以看到 Laravel 在取 Redis 要执行的队列的时候,同时会放一份到一个有序集合中,并使用过期时间戳作为分值。

只有当这个任务完成后,再把有序集合中这个任务移除。从这个有序集合移除队列的代码就省略,我们看一下 Laravel 如何处理执行时间大于 60 秒的队列。

也就是这段 lua 脚本执行的操作:

local val = redis.call('zrangebyscore', KEYS[1], '-inf', ARGV[1])if(next(val) ~= nil) then    redis.call('zremrangebyrank', KEYS[1], 0, #val - 1)    for i = 1, #val, 100 do        redis.call('rpush', KEYS[2], unpack(val, i, math.min(i+99, #val)))    endendreturn true

这里 zrangebyscore 找出分值从无限小到当前时间戳的元素,也就是 60 秒之前加入到集合的任务,然后通过 zremrangebyrank 从集合移除这些元素并 rpush 到队列中。

看到这里应该就恍然大悟了。

如果一个队列 60 秒没执行完,那么进程在取队列的时候从 reserved 集合中把这些任务又重新 rpush 到队列中。

以上是“Laravel会不会重复执行同一个队列任务”这篇文章的所有内容,感谢各位的阅读!希望分享的内容对大家有帮助,更多相关知识,欢迎关注编程网精选频道!

--结束END--

本文标题: Laravel会不会重复执行同一个队列任务

本文链接: https://lsjlt.com/news/275443.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作