Redis缓存的问题都是因为缓存过期,导致大量请求打到数据库,给数据库添加了压力。
以下是典型的三个缓存问题。
缓存穿透
概念
缓存穿透: 频繁请求缓存和数据库中没有的数据,导致数据库的压力过大
解决方案
- 规则校验:增加对key的规则校验,防止恶意请求
- 设默认值:数据库中没有数据时,给该key一个默认值,并设置合理的过期时间,防止较长时间的数据不一致
- 布隆过滤器:布隆过滤器中存在该key就通过,不存在就不访问Redis和数据库
基于设空值代码实现
public <R, ID> R queryWithPassThrough(String keyPrefix, ID id, Class<R> type, Function<ID, R> doFallback, Long time, TimeUnit timeUnit){
String key = keyPrefix + id;
String json = stringRedisTemplate.opsForValue().get(key);
// 如果缓存中有数据且不为空,直接返回
if (StrUtil.isNotBlank(json)){
return JSONUtil.toBean(json, type);
}
// 如果缓存中的数据为空,返回空值
if ("".equals(json)){
return null;
}
// 没有缓存,查询数据库
R result = doFallback.apply(id);
// 如果从数据库中查询不到数据,设空值
if (result == null){
stringRedisTemplate.opsForValue().set(key, "", time, timeUnit);
return null;
}
// 数据库中有数据
this.set(key, result, time, timeUnit);
return result;
}
缓存雪崩
概念
大量key同时过期或者Redis宕机,同时,大量请求打过来,导致数据库压力突然增大
解决方案
- 如果是宕机导致的雪崩,配置Redis集群,提高可用性
- 设置均匀分布的过期时间,防止同时过期,比如在5分钟的基础上,加上0-180秒的随机值
- 使用限流策略
- 服务降级,当Redis不可用时,进行服务降级,或者停用部分服务
缓存击穿
概念
大量请求同时访问一个过期的热key,导致数据库压力突然增大
解决方案
- 使用互斥锁:大量请求进来,只有一个请求能够拿到锁访问数据库,减少不必要的访问
- 设置永不过期:给热key不设置过期时间,或者开一个异步线程定时去更新热key的过期时间。
使用互斥锁解决
@Data
public class RedisData {
private LocalDateTime expireTime;
private Object data;
}
private static final ExecutorService CACHE_REBUILD_EXECUTOR = Executors.newFixedThreadPool(10);
// 设置逻辑过期
public void setWithLogicalExpire(String key, Object value, Long time, TimeUnit timeUnit){
// 设置逻辑过期
RedisData redisData = new RedisData();
redisData.setData(value);
redisData.setExpireTime(LocalDateTime.now().plusSeconds(timeUnit.toSeconds(time)));
// 写入Redis
stringRedisTemplate.opsForValue().set(key, JSONUtil.toJsonStr(redisData));
}
public boolean tryLock(String key){
Boolean flag = stringRedisTemplate.opsForValue().setIfAbsent(key, "1", 10, TimeUnit.MINUTES);
return BooleanUtil.isTrue(flag);
}
public void unlock(String key){
stringRedisTemplate.delete(key);
}
基于JVM的synchronized实现缓存击穿
// 基于synchronized互斥锁解决缓存击穿
public <R, ID> R queryWithLogicalExpire(String keyPrefix, ID id, Class<R> type, Function<ID, R> doFallback, Long time, TimeUnit timeUnit) {
// redis键
String key = keyPrefix + id;
// 从redis中查询缓存
String json = stringRedisTemplate.opsForValue().get(key);
// 判断是否有缓存 防止缓存穿透
if (StrUtil.isBlank(json)){
// 不存在
return null;
}
// 存在
RedisData redisData = JSONUtil.toBean(json, RedisData.class);
JSONObject data = (JSONObject) redisData.getData();
R result = JSONUtil.toBean(data, type);
// 判断是否过期
LocalDateTime expireTime = redisData.getExpireTime();
if (expireTime.isAfter(LocalDateTime.now())){
// 未过期,直接返回
return result;
}
// 缓存过期,需要缓存重建
String lockKey = "lock:" + id;
// 获取互斥锁
// 减小锁的粒度,锁的是每个键
synchronized (lockKey.intern()){
// 由于可能在这之前有人拿到锁并修改了,防止重复修改
// 双重检查锁
// 判断是否命中
// 1.从redis中查询缓存
String json2 = stringRedisTemplate.opsForValue().get(key);
if (StrUtil.isBlank(json2)){
// 不存在
return null;
}
// 存在
RedisData redisData2 = JSONUtil.toBean(json, RedisData.class);
JSONObject data2 = (JSONObject) redisData2.getData();
R result2 = JSONUtil.toBean(data2, type);
// 判断是否过期
LocalDateTime expireTime2 = redisData2.getExpireTime();
if (expireTime2.isAfter(LocalDateTime.now())){
// 未过期,直接返回
return result;
}
// 过期了,去数据库中查询新数据
R apply = doFallback.apply(id);
// 存入redis
this.setWithLogicalExpire(key, apply, time, timeUnit)
}
// 返回旧值
return result;
}
这种方式可以解决穿透问题,不过效率较低,会阻塞等待获取锁,降低了吞吐量。
基于Redis的SetNX实现
同时,我们可以根据redis的setnx的原子性来优化。
// 基于setnx互斥锁解决缓存击穿
public <R, ID> R queryWithLogicalExpire(String keyPrefix, ID id, Class<R> type, Function<ID, R> doFallback, Long time, TimeUnit timeUnit) {
// redis键
String key = keyPrefix + id;
// 从redis中查询缓存
String json = stringRedisTemplate.opsForValue().get(key);
// 判断是否有缓存 防止缓存穿透
if (StrUtil.isBlank(json)){
// 不存在
return null;
}
// 存在
RedisData redisData = JSONUtil.toBean(json, RedisData.class);
JSONObject data = (JSONObject) redisData.getData();
R result = JSONUtil.toBean(data, type);
// 判断是否过期
LocalDateTime expireTime = redisData.getExpireTime();
if (expireTime.isAfter(LocalDateTime.now())){
// 未过期,直接返回
return result;
}
// 缓存过期,需要缓存重建
String lockKey = "lock:" + id;
// 尝试获取锁
boolean isLock = tryLock(lockKey);
if (isLock){
// 由于可能在这之前有人拿到锁并修改了,防止重复修改
// 双重检查锁
// 判断是否命中
// 1.从redis中查询缓存
String json2 = stringRedisTemplate.opsForValue().get(key);
if (StrUtil.isBlank(json2)){
// 不存在
return null;
}
// 存在
RedisData redisData2 = JSONUtil.toBean(json, RedisData.class);
JSONObject data2 = (JSONObject) redisData2.getData();
R result2 = JSONUtil.toBean(data2, type);
// 判断是否过期
LocalDateTime expireTime2 = redisData2.getExpireTime();
if (expireTime2.isAfter(LocalDateTime.now())){
// 未过期,直接返回
return result2;
}
// 重建缓存
try {
// 查询数据库
R r1 = doFallback.apply(id);
// 写入redis
this.setWithLogicalExpire(key, r1, time, timeUnit);
} catch (Exception e) {
throw new RuntimeException(e);
} finally {
// 释放锁
unlock(lockKey);
}
}
// 返回旧值
return result;
}
如果获取锁失败,就直接返回上次的旧值,这种方式的效率较高,提高了系统的吞吐量,但是存在数据不一致的问题。
我们还可以对上述过程进行优化,将查询数据库和释放锁的过程交给一个新线程去做。
// 重建缓存
CACHE_REBUILD_EXECUTOR.submit(() -> {
// 重建缓存
try {
// 查询数据库
R r1 = doFallback.apply(id);
// 写入redis
this.setWithLogicalExpire(key, r1, time, timeUnit);
} catch (Exception e) {
throw new RuntimeException(e);
} finally {
// 释放锁
unlock(lockKey);
}
});
基于Redis的SetNX实现虽然效率较高,但是存在一个极端问题,在释放锁的过程中,有可能会误删他人的锁,因为释放锁的过程不是原子性的。
在线程1释放锁的过程中,恰好线程1cpu时间片用完,同时这个时候线程2拿到了锁,然后线程2cpu时间片用完,但是任务还没执行完,同时线程1得到新的时间片开始执行释放锁,这个时候就会把线程2的锁误删。