使用 spatie/laravel-unique-job 包可实现 Laravel 队列任务唯一性,通过 Redis 锁防止重复执行;也可手动在 handle 中加锁或分发前判断去重,推荐使用该包以确保稳定性和可靠性。

在 Laravel 中处理队列任务时,有时需要确保相同类型的任务不会重复执行,尤其是在高并发场景下。比如用户触发一个耗时操作,短时间内多次提交,如果不加控制,队列中就会堆积多个相同的任务,浪费资源甚至引发数据异常。Laravel 本身没有内置“唯一任务”的功能,但可以通过多种方式实现队列中的唯一任务(Unique Jobs)。
使用第三方包:laravel-unique-job
最常见且推荐的方式是使用社区广泛使用的扩展包 spatie/laravel-unique-job。它通过 Redis 锁机制防止相同任务重复入队或执行。
安装方法:
composer require spatie/laravel-unique-job使用示例:
在定义 Job 类时,引入 EnsureUnique trait,并实现 uniqueFor() 方法指定锁定时间:
use Spatie\UniqueJob\EnsureUnique;class SendWelcomeEmail implements ShouldQueue
{
use EnsureUnique;
// 锁定 10 分钟内不允许重复任务
public function uniqueFor(): int
{
return 60 * 10;
}
public function handle()
{
// 发送邮件逻辑
}
}
默认情况下,该任务的“唯一性”基于类名和参数。如果两个任务实例参数相同,第二个将被丢弃(可配置行为)。
自定义 Redis 锁实现
如果你不想引入第三方包,也可以手动使用 Redis 实现任务唯一性。
在 Job 的 handle() 方法开头尝试获取锁:
public function handle(){
$lockKey = 'send_welcome_email_' . $this->userId;
$locked = Cache::store('redis')->add($lockKey, true, 60); // 锁定 60 秒
if (! $locked) {
return; // 已存在执行中的任务,直接退出
}
try {
// 执行业务逻辑
} finally {
Cache::store('redis')->forget($lockKey); // 释放锁
}
}
这种方式灵活,但要注意异常情况下的锁释放,避免死锁。
在分发任务前做去重判断
另一种思路是在将任务推送到队列之前就判断是否已有相同任务待处理。
可以结合 Redis 记录“正在排队或执行中”的任务标识:
$jobId = 'welcome_email_' . $userId;if (! Cache::store('redis')->has($jobId)) {
Cache::store('redis')->put($jobId, true, 300); // 标记任务已入队
SendWelcomeEmail::dispatch($userId);
}
注意:这种方式不能 100% 保证唯一,因为任务执行完成后需清理标记,否则后续任务也无法进入。
总结与建议
实现 Laravel 队列中的唯一任务,关键是利用外部存储(如 Redis)做状态记录或加锁。推荐使用 spatie/laravel-unique-job 包,它封装良好、稳定可靠,适合大多数项目。
若项目已有 Redis 且需求简单,手动实现锁机制也是可行方案,但要特别注意锁的释放和超时设置。
基本上就这些,核心是避免重复任务占用系统资源,同时不影响正常业务流程。










