赞
踩
缓存穿透:key中对应的缓存数据不存在,导致去请求数据库,造成数据库的压力倍增的情况
缓存击穿:redis过期后的一瞬间,有大量用户请求同一个缓存数据,导致这些请求都去请求数据库,造成数据库压力倍增的情,针对一个key而言
缓存雪崩:缓存服务器宕机或者大量缓存集中某个时间段失效,导致请求全部去到数据库,造成数据库压力倍增的情况,这个是针对多个key而言
一、缓存穿透的解决方案
常用方法可以采用布隆过滤器方法进行数据拦截,其次可以还有一种解决思路,就是如果请求的数据为空,将空值也进行缓存,就不会发生穿透情况
<?php class getPrizeList { /** * redis实例 * @var \Redis */ private $redis; /** * @var string */ private $redis_key = '|prize_list'; /** * 过期时间 * @var int */ private $expire = 30; /** * getPrizeList constructor. * @param $redis */ public function __construct($redis) { $this->redis = $redis; } /** * @return array|bool|string */ public function fetch() { $result = $this->redis->get($this->redis_key); if(!isset($result)) { //此处应该进行数据库查询... //如果查询结果不存在,给其默认空数组进行缓存 $result = []; $this->redis->set($this->redis_key, $result, $this->expire); } return $result; } }
二、缓存击穿解决办法
使用互斥锁(mutex key),就是一个key过期时,多个请求过来允许其中一个请求去操作数据库,其他请求等待第一个请求成功返回结果后再请求。
<?php class getPrizeList { /** * redis实例 * @var \Redis */ private $redis; /** * @var string */ private $redis_key = '|prize_list'; /** * @var string */ private $setnx_key = '|prize_list_setnx'; /** * 过期时间 * @var int */ private $expire = 30; /** * getPrizeList constructor. * @param $redis */ public function __construct($redis) { $this->redis = $redis; } /** * @return array|bool|string */ public function fetch() { $result = $this->redis->get($this->redis_key); if(!isset($result)) { if($this->redis->setnx($this->setnx_key, 1, $this->expire)) { //此处应该进行数据库查询... //$result = 数据库查询结果; $this->redis->set($this->redis_key, $result, $this->expire); $this->redis->del($this->setnx_key); //删除互斥锁 } else { //其他请求每等待10毫秒重新请求一次 sleep(10); self::fetch(); } } return $result; } }
三、缓存雪崩的解决办法
<?php class getPrizeList { /** * redis实例 * @var \Redis */ private $redis; /** * @var string */ private $redis_key = '|prize_list'; /** * 缓存标记key * @var string */ private $cash_key = '|prize_list_cash'; /** * 过期时间 * @var int */ private $expire = 30; /** * getPrizeList constructor. * @param $redis */ public function __construct($redis) { $this->redis = $redis; } /** * @return array|bool|string */ public function fetch() { $cash_result = $this->redis->get($this->cash_key); $result = $this->redis->get($this->redis_key); if(!$cash_result) { $this->redis->set($this->cash_key, 1, $this->expire); //此处应该进行数据库查询... //$result = 数据库查询结果, 并且设置的时间要比cash_key长,这里设置为2倍; $this->redis->set($this->redis_key, $result, $this->expire * 2); $this->redis->del($this->cash_key); //删除互斥锁 } return $result; } }
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。