一、问题发生
正常情况下,通过 setnx 设置的一个锁,应该会1s后过期(防止并发请求,处理相同逻辑),但是实际发生的情况是,第一次处理下面的逻辑,就抛出异常。说明 setnx 函数返回了 false。
$redis_key_lock = 'lock';
//@todo 设置了一个1s的锁
$lock = $redis->setnx($redis_key_lock, 1);
//@todo 锁将会在1s后过期
$redis->expire($redis_key_lock, 1);
if ($lock) {
// 执行其他逻辑
} else {
throw new Exception("锁存在");
}
二、问题排查
redis 的内存占用满了
通过 redis.conf 查看一下 redis 过期策略
# maxmemory-policy volatile-lru
说明 redis 使用了默认的过期策略 noeviction
默认的过期策略 noeviction 不会对过期的 keys 做任何清理,当内存占用满后,会拒绝执行所有的写操作,通过翻阅redis配置文件,可以查到相关提示,并且除了 setnx,其他的写操作相关指令,都会执行失败。
解决方案就是快速释放 redis 内存,让 setnx 命令可以正常执行。



