laravel withoutOverlapping导致执行脚本延迟或者失效问题

时间:2021-07-15 18:59:48   收藏:0   阅读:0

2021年7月15日11:24:00

 

当执行某个需要经常跑的 everyMinute的之后防止,一次脚本未执行完就又重新跑,导致出现超出预期的错误,或者数据错误,就会使用 withoutOverlapping来防止重复执行

$schedule->command(‘NoPersonPrize‘)->everyMinute()->withoutOverlapping();

但是当你有多个withoutOverlapping和多个没有使用withoutOverlapping的放在Kernel的时候,会出现一些意外,比如有些脚本没有跑,或者延迟跑,是什么原因造成的呢?

 

 public function withoutOverlapping($expiresAt = 1440)
    {
        $this->withoutOverlapping = true;

        $this->expiresAt = $expiresAt;

        return $this->then(function () {
            $this->mutex->forget($this);
        })->skip(function () {
            return $this->mutex->exists($this);
        });
    }
withoutOverlapping的底层是利用缓存指定一个互斥的锁,如果你的laravel的缓存使用redis的时候,就会使用redis作为互斥锁的缓存key,默认缓存是file

$schedule->command(‘a‘)->everyMinute()->withoutOverlapping();

$schedule->command(‘b‘)->everyMinute();

$schedule->command(‘c‘)->everyMinute();

$schedule->command(‘d‘)->everyMinute()->withoutOverlapping();

 

这里多数人认为a,b,c,d会在每分钟开始会同事执行,其实不是,是先执行a,|(b,c)|d,是在a执行完在执行d,(b,c)是和a基本同时执行,

问题在于在互斥锁的问题,是执行一个完成之后在执行一个,不是每个脚本独立执行的

举例:

laravel_cache:framework/schedule-0e06f675a49cb9a0010858533b27d6a525caea88

内容:b:1;

技术图片

 

 

就会导致延迟执行某些脚本,甚至不执行
使用建议如下:
1,没有使用withoutOverlapping,在使用的上面,隔离开例如
$schedule->command(‘b‘)->everyMinute();
$schedule->command(‘c‘)->everyMinute();
#-------------------------------------------------------------------------
$schedule->command(‘a‘)->everyMinute()->withoutOverlapping();
$schedule->command(‘d‘)->everyMinute()->withoutOverlapping();

2,如果非必要每分钟执行的脚本,不要使用everyMinute+withoutOverlapping这种组合,避免重复执行
建立通过代码逻辑规避这个问题

3,如果多个有withoutOverlapping或者没有withoutOverlapping可以通过->dailyAt(‘18:33‘);
这中类似的特殊时间避开全部整点执行问题

4,减少或者不用withoutOverlapping


评论(0
© 2014 mamicode.com 版权所有 京ICP备13008772号-2  联系我们:gaon5@hotmail.com
迷上了代码!