Laravel中使用Redis锁解决缓存击穿问题
缓存击穿是指缓存中没有但数据库中有的数据(一般是缓存时间到期),这时由于并发用户特别多,同时读缓存没读到数据,又同时去数据库去取数据,引起数据库压力瞬间增大,造成过大压力。
Laravel 7 中自带有 \Illuminate\Cache\RedisLock Redis锁类,直接使用就行
使用 acquire() 方法获得互斥的Redis锁,使用 release() 方法释放锁(主要)
use Illuminate\Support\Facades\Redis;
use Illuminate\Cache\RedisLock;
function RedisLockTest(){
//获取redis实例
$redis = Redis::connection();
$key = 'redis_test_key';
//获取redis锁实例
$redisLock = new RedisLock($redis, $key . '_lock', 30);
$res = $redis->get($key);
if (empty($res)) {
//拿到互斥锁
if ($redisLock->acquire()) {
//模拟从数据库中获取数据的过程
sleep(5);
$value = date('Y-m-d H:i:s');
//更新缓存,过期时间可以根据实际情况调整
$redis->setex($key, 60, $value);
//释放锁
$redisLock->release();
return $value;
} else {
//等待2秒,然后重新获取缓存值,让其他获取到锁的进程取得数据并设置缓存,等待时间可以根据实际情况调整
sleep(2);
return $this->RedisLockTest();
}
} else {
return $res;
}}