小言_互联网的博客

Redis(开发与运维):57---缓存设计之(热点key重建优化)

441人阅读  评论(0)
  • 开发人员使用“缓存+过期时间”的策略既可以加速数据读写,又保证数据的定期更新,这种模式基本能够满足绝大部分需求。但是有两个问题如果同时出现,可能就会对应用造成致命的危害:
    • 当前key是一个热点key(例如一个热门的娱乐新闻),并发量非常 大
    • 重建缓存不能在短时间完成,可能是一个复杂计算,例如复杂的SQL、多次IO、多个依赖等
  • 在缓存失效的瞬间,有大量线程来重建缓存(如下图所示),造成 后端负载加大,甚至可能会让应用崩溃

  • 要解决这个问题也不是很复杂,但是不能为了解决这个问题给系统带来更多的麻烦,所以需要制定如下目标:
    • 减少重建缓存的次数
    • 数据尽可能一致
    • 较少的潜在危险

一、互斥锁(mutex)

  • 此方法只允许一个线程重建缓存,其他线程等待重建缓存的线程执行完,重新从缓存获取数据即可,整个过程如下图所示

  • 下面代码使用Redis的setnx命令实现上述功能:
    • 1)从Redis获取数据,如果值不为空,则直接返回值;否则执行下面的2.1)和2.2)步骤
    • 2.1)如果set(nx和ex)结果为true,说明此时没有其他线程重建缓存, 那么当前线程执行缓存构建逻辑
    • 2.2)如果set(nx和ex)结果为false,说明此时已经有其他线程正在执 行构建缓存的工作,那么当前线程将休息指定时间(例如这里是50毫秒,取 决于构建缓存的速度)后,重新执行函数,直到获取到数据

  
  1. String get(String key) {
  2. // 从Redis中获取数据
  3. String value = redis.get(key);
  4. // 如果value为空,则开始重构缓存
  5. if (value == null) {
  6. // 只允许一个线程重构缓存,使用nx,并设置过期时间ex
  7. String mutexKey = "mutext:key:" + key;
  8. if (redis.set(mutexKey, "1", "ex 180", "nx")) {
  9. // 从数据源获取数据
  10. value = db.get(key);
  11. // 回写Redis,并设置过期时间
  12. redis.setex(key, timeout, value);
  13. // 删除key_mutex
  14. redis.delete(mutexKey);
  15. }
  16. // 其他线程休息50毫秒后重试
  17. else {
  18. Thread.sleep( 50);
  19. get(key);
  20. }
  21. }
  22. return value;
  23. }

二、永远不过期

  • “永远不过期”包含两层意思:
    • 从缓存层面来看,确实没有设置过期时间,所以不会出现热点key过期 后产生的问题,也就是“物理”不过期
    • 从功能层面来看,为每个value设置一个逻辑过期时间,当发现超过逻 辑过期时间后,会使用单独的线程去构建缓存
  • 整个过程如下图所示:

  • 从实战看,此方法有效杜绝了热点key产生的问题,但唯一不足的就是重构缓存期间,会出现数据不一致的情况,这取决于应用方是否容忍这种不 一致
  • 下面代码使用Redis进行模拟:

  
  1. String get(final String key) {
  2. V v = redis.get(key);
  3. String value = v.getValue();
  4. // 逻辑过期时间
  5. long logicTimeout = v.getLogicTimeout();
  6. // 如果逻辑过期时间小于当前时间,开始后台构建
  7. if (v.logicTimeout <= System.currentTimeMillis()) {
  8. String mutexKey = "mutex:key:" + key;
  9. if (redis.set(mutexKey, "1", "ex 180", "nx")) {
  10. // 重构缓存
  11. threadPool.execute( new Runnable() {
  12. public void run() {
  13. String dbValue = db.get(key);
  14. redis.set(key, (dbvalue,newLogicTimeout));
  15. redis.delete(mutexKey);
  16. }
  17. });
  18. }
  19. }
  20. return value;
  21. }

三、总结

  • 作为一个并发量较大的应用,在使用缓存时有三个目标:
    • 第一,加快用户访问速度,提高用户体验
    • 第二,降低后端负载,减少潜在的风险,保证系统平稳
    • 第三,保证数据“尽可能”及时更新
  • 下面将按照这三个维度对上 述两种解决方案进行分析:

    • 互斥锁(mutex key):这种方案思路比较简单,但是存在一定的隐患,如果构建缓存过程出现问题或者时间较长,可能会存在死锁和线程池阻塞的风险,但是这种方法能够较好地降低后端存储负载,并在一致性上做得比较好
    • “永远不过期”:这种方案由于没有设置真正的过期时间,实际上已经 不存在热点key产生的一系列危害,但是会存在数据不一致的情况,同时代码复杂度会增大
  • 两种解决方法对比如下图所示:


转载:https://blog.csdn.net/qq_41453285/article/details/106555871
查看评论
* 以上用户言论只代表其个人观点,不代表本网站的观点或立场