设为首页 收藏本站
查看: 693|回复: 0

[经验分享] Redis之利用锁机制来防止缓存过期产生的惊群现象

[复制链接]

尚未签到

发表于 2018-11-4 13:53:22 | 显示全部楼层 |阅读模式
class KomaRedis{  
    private $redis; //redis对象
  
    private static $_instance = null;
  
    private function __construct($config = array())
  
    {
  
        if (empty($config)) {
  
            return false;
  
        }
  
        $this->redis = new Redis();
  
        $this->redis->connect($config['server'], $config['port']);
  
        return $this->redis;
  
    }
  
    /**
  
     * @param array $config
  
     * @return redis操作类对象
  
     */
  
    public static function getInstance($config = array())
  
    {
  
        if (!(self::$_instance instanceof self)) {
  
            self::$_instance = new self ($config);
  
        }
  
        return self::$_instance;
  
    }
  
    /**
  
     * 获取缓存
  
     * @param $key string $name
  
     * @return array,object,number,string,boolean
  
     * @desc 此方法使用了锁机制来防止防止缓存过期时所产生的惊群现象,保证只有一个进程不获取数据,可以更新,其他进程仍然获取过期数据
  
     */
  
    public function getByLock($key)
  
    {
  
        $sth = $this->redis->get($key);
  
        if ($sth === false) {
  
            return $sth;
  
        } else {
  
            $sth = json_decode($sth, TRUE);
  
            if (intval($sth['expire']) redis->incr($key . ".lock");
  
                if ($lock === 1) {
  
                    return false;
  
                } else {
  
                    return $sth['data'];
  
                }
  
            } else {
  
                return $sth['data'];
  
            }
  
        }
  
    }
  
    /**
  
     * 设置缓存
  
     * @param $key string $name 缓存键
  
     * @param $value $string ,array,object,number,boolean $value 缓存值
  
     * @param null $ttl $string ,number $ttl 过期时间,如果不设置,则使用默认时间,如果为 infinity 则为永久保存
  
     * @return bool
  
     * @desc 此方法存储的数据会自动加入一些其他数据来避免惊群现象,如需保存原始数据,请使用 set
  
     */
  
    public function setByLock($key, $value, $ttl = null)
  
    {
  
        if (is_numeric($ttl) && intval($ttl) > 0) {
  
            $ttl = intval($ttl);
  
            $exp = time() + $ttl;
  
            $arg = array("data" => $value, "expire" => $exp);
  
        } else {
  
            $ttl = 300;
  
            $exp = time() + $ttl;
  
        }
  
        empty($ttl) OR $ttl += 300; //增加redis缓存时间,使程序有足够的时间生成缓存
  
        $arg = array("data" => $value, "expire" => $exp);
  
        $rs = $this->redis->setex($key, $ttl, json_encode($arg, TRUE));
  
        $this->redis->del($key . ".lock");
  
        return $rs;
  
    }
  
    /**
  
     * 返回redis对象
  
     * redis有非常多的操作方法,我们只封装了一部分
  
     * 拿着这个对象就可以直接调用redis自身方法
  
     */
  
    public function redis()
  
    {
  
        return $this->redis;
  
    }
  
}



运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-630728-1-1.html 上篇帖子: redis主从同步与读写分离 下篇帖子: 搭建高可用及负载均衡的REDIS-dabao
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表