V2EX = way to explore
V2EX 是一个关于分享和探索的地方
现在注册
已注册用户请  登录
GeekHub
dyllen
V2EX  ›  PHP

PHP yii2 框架,定时任务加了 redis Mutex 还是重复执行了,怎么整?

  •  
  •   dyllen · 103 天前 · 835 次点击
    这是一个创建于 103 天前的主题,其中的信息可能已经有所发展或是发生改变。

    php

        public $mutexName = 'xxxxx';  //固定的值
    
        protected function checkThread()
        {
            \Yii::$app->setComponents([
                'mutex' => [
                    'class' => 'yii\redis\Mutex',
                    'expire' => 300,
                ],
            ]);
    
            return \Yii::$app->mutex->acquire($this->mutexName, 10);
        }
    

    有个定时任务,每次执行前我都会调用一下checkThread去获取锁,获取不到就退出。

    现在有多台机器设置了同样时刻的这个计划任务,在同一时间执行了这定时任务,我发现还是重复执行了后面的业务逻辑。

    这是这个获取锁没效?

    10 条回复    2020-06-19 09:32:17 +08:00
    ljwaheng
        1
    ljwaheng   103 天前
    这种是不是可以用 Redis 的 setnx 保证只有一个客户端获取锁? ( ex 设置过期时间)不过解锁的时候最好使用 Lua 解锁.
    dyllen
        2
    dyllen   103 天前
    @ljwaheng 你说的这些,`yii\redis\Mutex`里面都已经实现好了的。
    ljwaheng
        3
    ljwaheng   103 天前
    @dyllen #2 哦哦 没用过 yii2 。可是 Redis 不是单进程单线程吗?所以应该只有一个客户端获取到锁吧。
    surfire91
        4
    surfire91   103 天前
    猜测一个原因:
    锁是不是有实效?如果有实效的话,虽然你多台机器设置了同样时刻,但是不同机器时间可能存在误差,导致顺序执行了。比如 a 机器执行完了(锁销毁了),b 机器才开始执行任务(这个时候是能获取到锁到)。
    monsterxx03
        5
    monsterxx03   103 天前
    连到 redis 上, 用 monitor 指令看下这封装代码具体干了什么
    dyllen
        6
    dyllen   103 天前
    @surfire91 要是是你说的这情况,那就没问题。
    dyllen
        7
    dyllen   103 天前
    @monsterxx03 源码我都看过了,就两个操作,加锁 set nx px,解锁用的 lua,没了。
    surfire91
        8
    surfire91   103 天前
    @dyllen 别“要是”啊,验证下呗
    dyllen
        9
    dyllen   102 天前
    @surfire91 我是说你说的这个情况,不会有问题。顺序执行了,或者哪个先,那个后执行都不会有问题。
    这种锁本来就是执行完了就释放的。
    couashi
        10
    couashi   102 天前
    建议后期应该部署定时任务管理后台
    关于   ·   FAQ   ·   API   ·   我们的愿景   ·   广告投放   ·   感谢   ·   实用小工具   ·   3417 人在线   最高记录 5168   ·     Select Language
    创意工作者们的社区
    World is powered by solitude
    VERSION: 3.9.8.5 · 26ms · UTC 05:18 · PVG 13:18 · LAX 22:18 · JFK 01:18
    ♥ Do have faith in what you're doing.