总结两种常见的长列表分页缓存策略

简介: 通常,对于长列表加载的场景,都需要进行分页, 如最近的世界杯体育垂站项目中的赛程页,评论流,直播流。而为了提高分页加载的性能,往往需要对分页进行缓存。 下面总结对两种常见的分页缓存的策略, 适用场景以及各自的优缺点。     策略一: 直接对分页结果缓存 顾名思义,就是直接缓存每次分页查询的结果。  
通常,对于长列表加载的场景,都需要进行分页, 如最近的世界杯体育垂站项目中的赛程页,评论流,直播流。而为了提高分页加载的性能,往往需要对分页进行缓存。 下面总结对两种常见的分页缓存的策略, 适用场景以及各自的优缺点。
 
 

策略一: 直接对分页结果缓存

顾名思义,就是直接缓存每次分页查询的结果。
 
 
适用场景
  • 列表长度无法提前预知
  • 数据和排序不经常变更
 
优点:
  • 实现简单
  • 通常能获得不错的性能。由于直接缓存分页的结果,因此只需一次IO就能把数据查出来
 
缺点
  • 无法及时更新
随着列表增长,一个对象可能从第1页转移到第2页。因此当某个对象发生变更(或排序权重发生变化)后,无法判断该更新哪几页的缓存;除非同步维护一张倒排表,记录每个对象ID到所有缓存键值的映射关系, 但对于一般web应用实现成本有点高
 
  • 数据不一致
由于无法做到实时主动更新,因此失效时间不宜设置过长,这就需要根据实际业务场景(比如允许一定的更新延迟) 选取一个能接受的值, 而在缓存失效之前,需要忍受数据不一致。
 
  • 缓存键值(cacheKey)设计或使用不当, 可能会产生大量无效的缓存垃圾
假设分页查询的条件是 _prev_pos 和 size,_prev_pos为上一页最后一个对象的_pos值, 即从某个对象开始向前(或向后)检索size个对象, 则cacheKey = (_prev_pos+size), 选择不同的_prev_pos 和 size 会生成 不同的cacheKey。
类似的还有cacheKey= md5(url) , cacheKey = (startTime + endTime), 使用不当时也会产生大量垃圾cacheKey.
产生大量垃圾cacheKey的直接后果是,缓存空间会很快被耗尽,频繁触发LRU,最终影响应用的性能
 
 
 

策略二:缓存整个对象ID列表,再分别对每个对象建立缓存

 
预先把整个对象ID列表缓存起来,由程序实现分页逻辑,分页查询的结果是一个ID数组, 然后再根据每个ID从缓存中查找对象,最后组装成一个对象数组返回
 
适用场景
  • 列表长度有限
  • 对象数据或排序权重需要频繁更新
 
优点
  • 数据一致
在这种存储结构下, 当对象数据和排序权重发生变更时,能及时更新对应的缓存块,避免出现缓存的数据和数据库不一致的情况。又由于每次分页查询都是一次动态计算的结果,因此只要缓存更新了,就一定能拿到最新的结果
 
  • 缓存空间的大小是恒定且能提前预估
 
  • 缓存块能设置比较长的过期时间,不用担心缓存失效
 
缺点
  • IO次数 = n + 1 ( n 为每页的条数),为了保证性能, n 通常不能选得过大
  • 列表长度和 分页逻辑的算法 直接影响查询性能
  • 实现成本略高
 
 

策略3:缓存分页结果,并定时更新前几页缓存

在策略1的基础上,增加一个定时任务,定时刷新前几页的缓存, 从而尽量保证前几页的缓存是最新的。
 
由于在某些业务场景下,用户只会浏览前几页的内容,比如用户一般只会关注未来 1 ~ 2周的体育赛事, 因此只要保证前几页的内容是最新的即可。
 
 

总结

在项目实践中,我发现没有一种缓存策略能完美解决所有问题,往往需要在 性能 和 数据一致性之间寻找一个平衡点。 比如对于体育赛程列表,由于更新频率不高的特点,适合采用策略1 对赛程列表进行分页缓存,但是对于比赛直播流, 则采用策略2或策略3比较合适,特别对直播流有较强的人工运营需求(比如往流中插入一些竞猜题目或用户的评论).
 

 

目录
相关文章
|
3月前
|
缓存 算法 数据挖掘
深入理解缓存更新策略:从LRU到LFU
【10月更文挑战第7天】 在本文中,我们将探讨计算机系统中缓存机制的核心——缓存更新策略。缓存是提高数据检索速度的关键技术之一,无论是在硬件还是软件层面都扮演着重要角色。我们会详细介绍最常用的两种缓存算法:最近最少使用(LRU)和最少使用频率(LFU),并讨论它们的优缺点及适用场景。通过对比分析,旨在帮助读者更好地理解如何选择和实现适合自己需求的缓存策略,从而优化系统性能。
71 3
|
3月前
|
缓存 Java API
解密列表的创建与销毁,以及缓存池长什么样子?
解密列表的创建与销毁,以及缓存池长什么样子?
48 9
|
26天前
|
缓存 API C#
C# 一分钟浅谈:GraphQL 中的缓存策略
本文介绍了在现代 Web 应用中,随着数据复杂度的增加,GraphQL 作为一种更灵活的数据查询语言的重要性,以及如何通过缓存策略优化其性能。文章详细探讨了客户端缓存、网络层缓存和服务器端缓存的实现方法,并提供了 C# 示例代码,帮助开发者理解和应用这些技术。同时,文中还讨论了缓存设计中的常见问题及解决方案,如缓存键设计、缓存失效策略等,旨在提升应用的响应速度和稳定性。
41 13
|
5月前
|
缓存 Java
Java本地高性能缓存实践问题之Caffeine缓存库中基于时间设置驱逐策略的问题如何解决
Java本地高性能缓存实践问题之Caffeine缓存库中基于时间设置驱逐策略的问题如何解决
|
9天前
|
存储 消息中间件 设计模式
缓存数据一致性策略如何分类?
数据库与缓存数据一致性问题的解决方案主要分为强一致性和最终一致性。强一致性通过分布式锁或分布式事务确保每次写入后数据立即一致,适合高要求场景,但性能开销大。最终一致性允许短暂延迟,常用方案包括Cache-Aside(先更新DB再删缓存)、Read/Write-Through(读写穿透)和Write-Behind(异步写入)。延时双删策略通过两次删除缓存确保数据最终一致,适用于复杂业务场景。选择方案需根据系统复杂度和一致性要求权衡。
36 0
|
2月前
|
存储 缓存 安全
在 Service Worker 中配置缓存策略
Service Worker 是一种可编程的网络代理,允许开发者控制网页如何加载资源。通过在 Service Worker 中配置缓存策略,可以优化应用性能,减少加载时间,提升用户体验。此策略涉及缓存的存储、更新和检索机制。
|
2月前
|
存储 缓存 监控
利用 Redis 缓存特性避免缓存穿透的策略与方法
【10月更文挑战第23天】通过以上对利用 Redis 缓存特性避免缓存穿透的详细阐述,我们对这一策略有了更深入的理解。在实际应用中,我们需要根据具体情况灵活运用这些方法,并结合其他技术手段,共同保障系统的稳定和高效运行。同时,要不断关注 Redis 缓存特性的发展和变化,及时调整策略,以应对不断出现的新挑战。
74 10
|
2月前
|
Web App开发 缓存 UED
如何设置浏览器的缓存策略?
【10月更文挑战第23天】通过合理地设置浏览器的缓存策略,可以在提高网页性能、减少网络流量的同时,确保用户能够获取到最新的内容,从而提升用户体验和网站的性能优化效果。
117 4
|
2月前
|
存储 消息中间件 缓存
缓存策略
【10月更文挑战第25天】在实际应用中,还需要不断地监控和调整缓存策略,以适应系统的变化和发展。
|
2月前
|
缓存 监控 NoSQL
Redis 缓存穿透及其应对策略
【10月更文挑战第23天】通过以上对 Redis 缓存穿透的详细阐述,我们对这一问题有了更深入的理解。在实际应用中,我们需要根据具体情况综合运用多种方法来解决缓存穿透问题,以保障系统的稳定运行和高效性能。同时,要不断关注技术的发展和变化,及时调整策略,以应对不断出现的新挑战。
59 4