Redis使用细节
分布式锁
因为Redis是单线程的,所以可以用setnx来模拟锁的获取释放从而实现分布式锁
在用setnx实现分布式锁时,会出现一些问题
- 业务超时解锁,导致并发问题。业务执行时间超过了锁超时的时间
- redis主从切换临界点问题,主从切换后,A持有的锁还没有同步到新的主节点,B在新的主节点获取到了锁
- redis集群脑裂,导致出现多个主节点
大key和热key
大key定义
- String类型:value的字节数大于10KB即为大Key
- Hash/Set等复杂结构类型:元素个数大于5000个或总value字节数大于10M即为大key
大key危害
- 读取成本高
- 容易导致慢查询
- 主从复制异常,服务阻塞,无法正常响应请求
消除大key的方法
1.拆分
将大key拆分为小key,例如一个String拆分成多个String
2.压缩
将value压缩后写入Redis,读取时解压后再用,注意选择一个合适的算法
3.集合类结构hash消除的方法
- 拆分,用hash取余,位掩码的方式决定放在哪个key中
- 区分冷热:比如榜单列表场景使用zset,只缓存前10页数据,后续的数据走db
热key定义
用户访问一个key的QPS特别高,导致server出现CPU负载突增或者不均的情况,热key没有明确的标准,一般QPS超过500就有可能被识别为热Key
解决热Key的方法
1.设置Localcache
在访问Redis前,在业务侧设置Localcache,降低访问Redis的QPS,如果Localcache过期或者未命中,则从Redis中将数据更新到LocalCache
2.拆分
将一个热key复制写入多份,访问的时候访问多个key,但是value是同一个,但是代价是更新时需要更新多个key
慢查询场景
容易导致慢查询的操作
- 批量操作一次性传入过多的key/value,如mset/hmset等操作,建议单批次不要超过100
- zset大部分命令都是O(logn),当大小超过5k以上时,简单的zadd/zerm也可能导致慢查询
- 操作大key
缓存穿透和缓存雪崩
缓存穿透:热点数据查询绕过缓存,直接查询数据库
缓存雪崩:大量缓存同时过期
缓存穿透的危害:
- 查询一个一定不存在的数据,这样的所有请求都会打到db上
- 缓存过期时,一个热key过期,也会有大量的请求同时击穿至db
减少缓存穿透的方法
- 缓存空值,在查询到在缓存和数据库中都不存在,则可以缓存一个空值
- 布隆过滤器,使用一个算法来存储合法key
避免缓存雪崩的方法
- 将缓存失效的时间分散开,比如在原有的失效时间基础上增加一个随机值
- 使用缓存集群,避免单机宕机造成的缓存雪崩
缓存击穿问题
缓存击穿也叫热点key问题,被高并发访问以及缓存重建业务较复杂的key突然失效了,会对数据库带来巨大冲击。
解决方法:
互斥锁和逻辑过期
1.互斥锁就是在做缓存重建时进行加锁,其余线程进行带有休眠的重试
缺点:
线程需要等待,性能受影响
会有死锁风险
2.逻辑过期,不设置真正的过期时间,而是设置一个字段作为逻辑过期
获取互斥锁后开启一个新线程,进行缓存重建,同时原始线程返回一个旧的信息
其余线程发现过期后获取互斥锁失败后不进行等待,直接也返回旧数据
缺点:
不能保证一致性
数据库和缓存如何保持一致性
无论是先更新数据库还是先更新缓存,都有可能无法保证数据库的一致性
于是引出旁路缓存策略(cache aside pattern)
更新数据时,不更新缓存,而是删除缓存中的数据。然后到读取数据时,发现缓存中没有数据之后,再从数据库中读取数据,更新到缓存中
该策略可以分为两个策略,读策略和写策略
写策略是更新数据库或删除缓存
读策略是缓存未命中后读取数据库的数据和回写数据
先删除缓存,再更新数据库会造成数据不一致的问题
而先更新数据库,再删除缓存虽然也可能会存在数据不一致的问题
但是因为写缓存的速度远远比写数据库的速度要快,所以缓存不一致的情况很少发生
所以,最好的方案就是先更新数据库,再删除缓存
删除相对于更新的好处除了数据一致性的问题,还因为每次更新数据库都更新缓存,无效写比较多。
然后给缓存数据加上过期时间,防止意外的缓存与数据库的数据不一致
但是该方法在每次更新数据库时,缓存的数据会被删除,这样会对缓存的命中率带来影响
除了旁路更新策略以外
还有write behind caching pattern
调用者只操作缓存,由其它线程异步的将缓存数据持久化到数据库,保证最终一致
不过目前主流的还是旁路缓存策略
然后我们还要保证数据库与缓存操作的原子性
在单体系统里,将缓存和数据库操作放在同一个事务
如果不使用事务的话,要保证操作一致性,可以采用重试机制
Redis分布式锁误删问题
在使用的时候设置值为该线程所产生的唯一的一个id,当这个线程阻塞后导致锁过期时,新的线程再重新加锁,原来的线程执行完毕之后判断当前锁是不是本线程产生的锁,如果不是则就不删除,这样就防止误删问题,不过为了加锁时间长于业务执行的时间,我们需要别的解决方法
Redission分布式锁
这里重点讲解一下ReentrantLock,可重入锁
解决锁的重入以及重试问题,并且防止业务用时超过加锁时间
加锁和解锁是用Lua脚本原子性的实现的
锁的可重试
先尝试获取锁,然后申请锁的耗时如果大于等于最大等待时间,则申请锁失败
可以订阅锁释放事件,通过awit方法阻塞等待锁释放,防止无效的锁资源申请问题,一旦锁释放会发消息通知等待的线程进行竞争
锁的续期机制
Redission提供了一个续期机制,只要一旦加锁成功,就会启动一个watchdog,每隔一段时间就会检查一下是否持有锁,如果持有,就会延长锁的时间
缺点就是采用多个Redis时会出现一些问题
锁的可重入机制
多记录一个加锁次数,每次访问使得锁的重入次数加1