如何进行缓存?Redis 的常见数据结构与用例

答案:Redis通过缓存旁路模式提升系统性能,利用String、Hash、List、Set、Sorted Set等数据结构适配不同场景,结合TTL、主动失效、分布式锁等策略保障数据一致性与高并发,需综合考虑命中率、一致性、缓存容量及穿透、雪崩、击穿等问题,实现高效稳定的缓存体系。

如何进行缓存?redis 的常见数据结构与用例

缓存,说白了,就是把那些访问频率高、计算成本大的数据提前存起来,等下次再用的时候直接拿,不用重新计算或查询,大大提升响应速度。它就像你常用的工具箱,把最常用的工具放在触手可及的地方,省去了每次都去仓库里找的麻烦。而 Redis,以其内存级的操作速度和多样化的数据结构,无疑是实现高效缓存的首选利器,它不仅快,还能干很多别的事。

解决方案

进行缓存的核心思路其实很简单:在数据被请求时,先去缓存里找。如果找到了(命中),就直接返回;如果没找到(未命中),再去原始数据源(比如数据库)获取,拿到数据后,在返回给请求方的同时,也把这份数据存一份到缓存里,以便下次使用。这个过程通常被称为“缓存旁路”(Cache-Aside)模式,也是我们最常用的一种。

在实际操作中,这通常意味着你的应用代码会在每次需要数据时,先调用 Redis 客户端去查询。例如,当用户请求一个商品详情页,我们不会每次都去数据库查询商品的完整信息。我们会先检查 Redis 里有没有这个商品的 ID 对应的详情数据。如果有,直接展示;如果没有,才去数据库捞,捞出来后,顺手就往 Redis 里塞一份,可能还会给它设置一个过期时间(TTL),这样数据就不会永远占着缓存空间,也避免了长时间的数据不一致。

但缓存的艺术远不止于此,它涉及到数据一致性、过期策略、淘汰机制等一系列考量。比如说,当数据库中的原始数据发生变化时,缓存中的旧数据就成了“脏数据”,这时就需要一个策略来让缓存失效,或者更新缓存。这通常可以通过主动删除缓存条目,或者设置较短的过期时间来解决。有时候,我们还会遇到缓存雪崩、缓存穿透、缓存击穿等问题,这些都需要在设计时就有所考虑,并采取相应的预防措施。

Redis 的常见数据结构有哪些,它们各自适合什么场景?

Redis 之所以强大,很大程度上是因为它提供了远超普通键值对存储的丰富数据结构。我个人觉得,理解这些结构是玩转 Redis 的基础。

String (字符串):这是最基础的键值对,一个键对应一个字符串值。这个值可以是文本、数字,甚至是二进制数据。

适用场景:最常见的,比如存储用户的会话 token、文章的访问量、某个配置项的值。比如,你可以用

SET user:1:name "张三"

来存储用户名,用

INCR article:123:views

来统计文章阅读量。它简单直接,几乎无处不在。我的看法:虽然简单,但它是构建其他复杂功能的基础。很多时候,我们不需要复杂的结构,一个字符串就够了。

Hash (哈希):一个键对应一个哈希表,哈希表里可以存储多个字段和值。你可以把它想象成一个对象或者一个字典。

适用场景:存储对象的属性。比如,存储用户信息

user:1

name

age

email

等字段。

HSET user:1 name "李四" age 30 email "li@example.com"

。这比用多个 String 键来存储一个对象的属性要高效得多,也更易于管理。我的看法:非常适合存储结构化数据,比如用户资料、商品信息。通过一个键就能获取或修改对象的某个属性,减少了网络开销。

List (列表):一个键对应一个列表,列表是按照插入顺序排序的字符串元素集合。你可以从列表的两端推入或弹出元素。

适用场景:消息队列(简单的)、最新动态、关注者列表。比如,你可以用

LPUSH timeline:user:1 "新动态1"

来添加用户动态,

LRANGE timeline:user:1 0 9

来获取最新十条动态。我的看法:对于需要维护顺序,或者实现生产者/消费者模式(比如延迟队列)的场景非常有用。

BLPOP

/

BRPOP

命令在构建阻塞队列时简直是神器。

Set (集合):一个键对应一个无序的字符串元素集合,集合中的元素是唯一的,不允许重复。

适用场景:存储标签、共同关注、抽奖活动参与者。比如,

SADD tags:article:1 "技术" "Redis"

SINTER

可以找到共同关注的人,

SRANDMEMBER

可以随机抽取中奖者。我的看法:当你需要快速判断一个元素是否存在于某个集合中,或者进行集合间的交集、并集、差集运算时,Set 是不二之选。

Sorted Set (有序集合):一个键对应一个有序集合,与 Set 类似,但每个元素都会关联一个浮点数分数(score),集合中的元素会按照分数从小到大排序。分数相同的元素则按字典序排序。

适用场景:排行榜、带权重的标签、最近活跃用户。比如,

ZADD leaderboard 100 "playerA"

ZREVRANGE leaderboard 0 9 WITHSCORES

可以获取前十名玩家及分数。我的看法:这是我个人觉得 Redis 最具“杀手级”功能的数据结构之一。构建实时排行榜简直是它的拿手好戏,效率高得惊人。

在实际项目中,我们通常如何利用 Redis 实现高效缓存?

在实际项目中,利用 Redis 实现高效缓存不仅仅是“存取数据”那么简单,它更像是一套组合拳,需要根据业务场景和数据特性来选择合适的策略和数据结构。

最核心的当然是前面提到的 “缓存旁路”模式。当应用需要数据时,它首先会尝试从 Redis 中获取。如果 Redis 中有,直接返回;如果没有,就去数据库查询,然后将查询结果写入 Redis,并设置一个合适的过期时间(TTL)。这个 TTL 的设置非常关键,它决定了数据在缓存中停留多久,既要保证一定的新鲜度,又要避免缓存失效过快导致频繁回源。

除了单纯的数据缓存,Redis 还被广泛用于以下场景,它们也间接提升了系统的“缓存”能力:

会话缓存 (Session Caching):在分布式系统中,用户登录后的会话信息(如用户ID、权限等)如果存储在应用服务器本地,会导致用户在不同服务器间跳转时会话丢失。将这些会话信息存储在 Redis 中,所有应用服务器都可以共享,实现了会话的集中管理和快速访问。这就像把用户的“身份卡”放在一个所有人都认识的公共柜台,而不是每个接待员都发一张。全页缓存 (Full Page Caching):对于那些不经常变化、但访问量巨大的页面(如新闻首页、商品分类页),可以直接将整个 HTML 页面内容作为字符串存储在 Redis 中。当用户请求这些页面时,Web 服务器可以直接从 Redis 中获取并返回,绕过了应用服务器的渲染过程,极大地提升了响应速度和并发能力。对象缓存 (Object Caching):将数据库查询结果集、复杂计算的结果或者序列化后的对象存储在 Redis 中。例如,一个复杂的报表数据可能需要聚合多个表才能生成,将其结果缓存起来,可以避免每次请求都进行耗时的计算。计数器与限流 (Counters & Rate Limiting):利用 Redis 的原子操作(如

INCR

),可以非常高效地实现各种计数器,比如文章阅读量、点赞数。结合过期时间,还可以实现接口的访问频率限制(限流),防止恶意请求或系统过载。比如,限制一个IP地址每秒只能访问某个接口N次。分布式锁 (Distributed Locks):在分布式环境中,为了保证某个操作的原子性,防止多个进程同时修改同一份资源,Redis 可以用来实现分布式锁。这本质上也是一种“状态缓存”,通过一个键的存在与否来表示锁的状态。

在实践中,我们还需要关注缓存的更新策略。除了 TTL,当后端数据发生变化时,我们往往需要主动去删除或更新 Redis 中的对应缓存项,以保证数据一致性。例如,商品信息更新后,需要通过消息队列通知所有相关的缓存失效。这比等待 TTL 过期更及时,但增加了系统的复杂性。

缓存策略的选择与优化,有哪些关键考量点?

选择和优化缓存策略,绝不是一拍脑袋就能决定的事情,它需要深入理解业务特性、数据访问模式以及系统架构。这里面有一些我个人觉得非常关键的考量点:

命中率 (Cache Hit Ratio):这是衡量缓存效果最直观的指标。命中率高,说明大部分请求都被缓存处理了,系统性能自然好。优化缓存策略,很大程度上就是想办法提高命中率。这通常意味着你需要缓存那些访问频率最高的数据,并确保缓存的容量足够容纳这些“热点”数据。如果命中率持续低迷,那可能你的缓存策略有问题,或者根本不适合缓存。

数据一致性与新鲜度 (Consistency vs. Freshness):这是缓存领域永恒的难题。缓存的存在本身就引入了数据不一致的风险。是选择强一致性(数据永远最新,但性能可能受影响),还是选择最终一致性(数据可能短暂过期,但性能极高)?这取决于你的业务场景。对于金融交易这类对数据一致性要求极高的场景,可能需要更复杂的缓存更新机制,甚至不适合缓存。而对于新闻阅读、商品展示这类对实时性要求不那么严苛的场景,接受一定程度的延迟是完全可以的,可以采用更宽松的过期策略。我倾向于在保证业务可接受的前提下,尽可能偏向最终一致性,以换取更好的性能。

缓存失效策略 (Cache Invalidation Strategy):如何让缓存中的旧数据失效,是缓存设计中最复杂的部分。

TTL (Time To Live):最简单直接,给缓存项设置一个过期时间。时间一到,自动失效。适用于数据变化不频繁或对实时性要求不高的场景。LRU (Least Recently Used):当缓存空间不足时,淘汰最近最少使用的数据。Redis 默认支持多种淘汰策略,LRU 是最常用的一种。LFU (Least Frequently Used):淘汰最不经常使用的数据。这对于那些在某个时间段内很热门,但之后就很少被访问的数据,比 LRU 更有效。主动删除/更新:当后端数据发生变化时,通过应用代码主动删除或更新 Redis 中的缓存项。这是最能保证数据一致性的方式,但实现起来也最复杂,需要设计事件通知机制(如消息队列)。

缓存容量与成本 (Capacity & Cost):Redis 是内存数据库,内存成本相对较高。你需要合理评估需要缓存的数据量,以及你愿意为之支付的成本。不要盲目地把所有数据都塞进 Redis。对于超大数据量,可能需要考虑 Redis Cluster 进行分片,或者结合本地缓存、CDN 等多级缓存策略。

缓存穿透、雪崩与击穿 (Cache Penetration, Avalanche & Breakdown)

穿透:查询一个根本不存在的数据,缓存和数据库都没有。攻击者可能会利用这一点,持续查询不存在的数据,导致数据库压力过大。通常通过布隆过滤器或者缓存空值来解决。雪崩:大量缓存同时失效,导致所有请求直接打到数据库,数据库瞬间崩溃。可以通过设置不同的过期时间、使用 Redis Cluster 或多级缓存来缓解。击穿:某个热点数据突然失效,大量请求同时去查询这个数据,导致数据库压力过大。可以通过互斥锁(只允许一个请求去数据库查询,其他请求等待)或者永不过期(后台异步更新)来解决。

监控与告警 (Monitoring & Alerting):任何缓存系统都需要完善的监控。你需要关注 Redis 的内存使用率、CPU 使用率、QPS(每秒查询数)、命中率、连接数等关键指标。一旦发现异常,能及时告警并介入处理。

在我看来,缓存策略没有“银弹”,只有“最适合”。在设计之初,就应该充分考虑业务场景、数据特性和技术栈,并在系统运行过程中持续观察、调整和优化。这更像是一门平衡的艺术,在性能、成本和数据一致性之间找到最佳的平衡点。

以上就是如何进行缓存?Redis 的常见数据结构与用例的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1369906.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月14日 10:04:09
下一篇 2025年12月14日 10:04:22

相关推荐

  • Uniapp 中如何不拉伸不裁剪地展示图片?

    灵活展示图片:如何不拉伸不裁剪 在界面设计中,常常需要以原尺寸展示用户上传的图片。本文将介绍一种在 uniapp 框架中实现该功能的简单方法。 对于不同尺寸的图片,可以采用以下处理方式: 极端宽高比:撑满屏幕宽度或高度,再等比缩放居中。非极端宽高比:居中显示,若能撑满则撑满。 然而,如果需要不拉伸不…

    2025年12月24日
    400
  • 如何让小说网站控制台显示乱码,同时网页内容正常显示?

    如何在不影响用户界面的情况下实现控制台乱码? 当在小说网站上下载小说时,大家可能会遇到一个问题:网站上的文本在网页内正常显示,但是在控制台中却是乱码。如何实现此类操作,从而在不影响用户界面(UI)的情况下保持控制台乱码呢? 答案在于使用自定义字体。网站可以通过在服务器端配置自定义字体,并通过在客户端…

    2025年12月24日
    800
  • 如何在地图上轻松创建气泡信息框?

    地图上气泡信息框的巧妙生成 地图上气泡信息框是一种常用的交互功能,它简便易用,能够为用户提供额外信息。本文将探讨如何借助地图库的功能轻松创建这一功能。 利用地图库的原生功能 大多数地图库,如高德地图,都提供了现成的信息窗体和右键菜单功能。这些功能可以通过以下途径实现: 高德地图 JS API 参考文…

    2025年12月24日
    400
  • 如何使用 scroll-behavior 属性实现元素scrollLeft变化时的平滑动画?

    如何实现元素scrollleft变化时的平滑动画效果? 在许多网页应用中,滚动容器的水平滚动条(scrollleft)需要频繁使用。为了让滚动动作更加自然,你希望给scrollleft的变化添加动画效果。 解决方案:scroll-behavior 属性 要实现scrollleft变化时的平滑动画效果…

    2025年12月24日
    000
  • 如何为滚动元素添加平滑过渡,使滚动条滑动时更自然流畅?

    给滚动元素平滑过渡 如何在滚动条属性(scrollleft)发生改变时为元素添加平滑的过渡效果? 解决方案:scroll-behavior 属性 为滚动容器设置 scroll-behavior 属性可以实现平滑滚动。 html 代码: click the button to slide right!…

    2025年12月24日
    500
  • 如何选择元素个数不固定的指定类名子元素?

    灵活选择元素个数不固定的指定类名子元素 在网页布局中,有时需要选择特定类名的子元素,但这些元素的数量并不固定。例如,下面这段 html 代码中,activebar 和 item 元素的数量均不固定: *n *n 如果需要选择第一个 item元素,可以使用 css 选择器 :nth-child()。该…

    2025年12月24日
    200
  • 使用 SVG 如何实现自定义宽度、间距和半径的虚线边框?

    使用 svg 实现自定义虚线边框 如何实现一个具有自定义宽度、间距和半径的虚线边框是一个常见的前端开发问题。传统的解决方案通常涉及使用 border-image 引入切片图片,但是这种方法存在引入外部资源、性能低下的缺点。 为了避免上述问题,可以使用 svg(可缩放矢量图形)来创建纯代码实现。一种方…

    2025年12月24日
    100
  • 旋转长方形后,如何计算其相对于画布左上角的轴距?

    绘制长方形并旋转,计算旋转后轴距 在拥有 1920×1080 画布中,放置一个宽高为 200×20 的长方形,其坐标位于 (100, 100)。当以任意角度旋转长方形时,如何计算它相对于画布左上角的 x、y 轴距? 以下代码提供了一个计算旋转后长方形轴距的解决方案: const x = 200;co…

    2025年12月24日
    000
  • 旋转长方形后,如何计算它与画布左上角的xy轴距?

    旋转后长方形在画布上的xy轴距计算 在画布中添加一个长方形,并将其旋转任意角度,如何计算旋转后的长方形与画布左上角之间的xy轴距? 问题分解: 要计算旋转后长方形的xy轴距,需要考虑旋转对长方形宽高和位置的影响。首先,旋转会改变长方形的长和宽,其次,旋转会改变长方形的中心点位置。 求解方法: 计算旋…

    2025年12月24日
    000
  • 旋转长方形后如何计算其在画布上的轴距?

    旋转长方形后计算轴距 假设长方形的宽、高分别为 200 和 20,初始坐标为 (100, 100),我们将它旋转一个任意角度。根据旋转矩阵公式,旋转后的新坐标 (x’, y’) 可以通过以下公式计算: x’ = x * cos(θ) – y * sin(θ)y’ = x * …

    2025年12月24日
    000
  • 如何让“元素跟随文本高度,而不是撑高父容器?

    如何让 元素跟随文本高度,而不是撑高父容器 在页面布局中,经常遇到父容器高度被子元素撑开的问题。在图例所示的案例中,父容器被较高的图片撑开,而文本的高度没有被考虑。本问答将提供纯css解决方案,让图片跟随文本高度,确保父容器的高度不会被图片影响。 解决方法 为了解决这个问题,需要将图片从文档流中脱离…

    2025年12月24日
    000
  • 如何计算旋转后长方形在画布上的轴距?

    旋转后长方形与画布轴距计算 在给定的画布中,有一个长方形,在随机旋转一定角度后,如何计算其在画布上的轴距,即距离左上角的距离? 以下提供一种计算长方形相对于画布左上角的新轴距的方法: const x = 200; // 初始 x 坐标const y = 90; // 初始 y 坐标const w =…

    2025年12月24日
    200
  • CSS元素设置em和transition后,为何载入页面无放大效果?

    css元素设置em和transition后,为何载入无放大效果 很多开发者在设置了em和transition后,却发现元素载入页面时无放大效果。本文将解答这一问题。 原问题:在视频演示中,将元素设置如下,载入页面会有放大效果。然而,在个人尝试中,并未出现该效果。这是由于macos和windows系统…

    2025年12月24日
    200
  • 为什么 CSS mask 属性未请求指定图片?

    解决 css mask 属性未请求图片的问题 在使用 css mask 属性时,指定了图片地址,但网络面板显示未请求获取该图片,这可能是由于浏览器兼容性问题造成的。 问题 如下代码所示: 立即学习“前端免费学习笔记(深入)”; icon [data-icon=”cloud”] { –icon-cl…

    2025年12月24日
    200
  • 如何利用 CSS 选中激活标签并影响相邻元素的样式?

    如何利用 css 选中激活标签并影响相邻元素? 为了实现激活标签影响相邻元素的样式需求,可以通过 :has 选择器来实现。以下是如何具体操作: 对于激活标签相邻后的元素,可以在 css 中使用以下代码进行设置: li:has(+li.active) { border-radius: 0 0 10px…

    2025年12月24日
    100
  • 如何模拟Windows 10 设置界面中的鼠标悬浮放大效果?

    win10设置界面的鼠标移动显示周边的样式(探照灯效果)的实现方式 在windows设置界面的鼠标悬浮效果中,光标周围会显示一个放大区域。在前端开发中,可以通过多种方式实现类似的效果。 使用css 使用css的transform和box-shadow属性。通过将transform: scale(1.…

    2025年12月24日
    200
  • 如何计算旋转后的长方形在画布上的 XY 轴距?

    旋转长方形后计算其画布xy轴距 在创建的画布上添加了一个长方形,并提供其宽、高和初始坐标。为了视觉化旋转效果,还提供了一些旋转特定角度后的图片。 问题是如何计算任意角度旋转后,这个长方形的xy轴距。这涉及到使用三角学来计算旋转后的坐标。 以下是一个 javascript 代码示例,用于计算旋转后长方…

    2025年12月24日
    000
  • 为什么我的 Safari 自定义样式表在百度页面上失效了?

    为什么在 Safari 中自定义样式表未能正常工作? 在 Safari 的偏好设置中设置自定义样式表后,您对其进行测试却发现效果不同。在您自己的网页中,样式有效,而在百度页面中却失效。 造成这种情况的原因是,第一个访问的项目使用了文件协议,可以访问本地目录中的图片文件。而第二个访问的百度使用了 ht…

    2025年12月24日
    000
  • 如何用前端实现 Windows 10 设置界面的鼠标移动探照灯效果?

    如何在前端实现 Windows 10 设置界面中的鼠标移动探照灯效果 想要在前端开发中实现 Windows 10 设置界面中类似的鼠标移动探照灯效果,可以通过以下途径: CSS 解决方案 DEMO 1: Windows 10 网格悬停效果:https://codepen.io/tr4553r7/pe…

    2025年12月24日
    000
  • 使用CSS mask属性指定图片URL时,为什么浏览器无法加载图片?

    css mask属性未能加载图片的解决方法 使用css mask属性指定图片url时,如示例中所示: mask: url(“https://api.iconify.design/mdi:apple-icloud.svg”) center / contain no-repeat; 但是,在网络面板中却…

    2025年12月24日
    000

发表回复

登录后才能评论
关注微信