如何使用 Redis 作为高效缓存

devtools/2025/1/21 7:21:54/

如何使用 Redis 作为高效缓存

Redis(Remote Dictionary Server)是一个高性能的 内存存储系统,通常被用作 缓存 来加速数据访问,提高应用的吞吐量和响应速度。本文详细讲解如何使用 Redis 作为高效缓存,包括基本原理、常见模式、最佳实践以及优化技巧。


1. 为什么使用 Redis 作为缓存

相比于传统的数据库,Redis 具有以下优点:

  • 低延迟 & 高吞吐:Redis 基于内存操作,读写速度远超磁盘存储数据库
  • 支持多种数据结构:支持 StringHashListSetSorted Set 等丰富的数据类型,适合不同的缓存场景。
  • 持久化支持:可选择性地使用 AOF 和 RDB 进行数据持久化,防止数据丢失。
  • 分布式支持:支持主从复制、哨兵模式和集群模式,能够横向扩展。
  • 丰富的过期策略:支持多种缓存淘汰策略,避免缓存占用过多内存。

2. Redis 缓存的常见使用模式

Redis 作为缓存一般采用 前置缓存(Look-aside Cache)写穿透缓存(Write-through Cache) 模式。

2.1. 前置缓存(Look-aside Cache)

原理:

  1. 先查询 Redis 缓存,如果命中则直接返回;
  2. 如果未命中(Cache Miss),则查询数据库,并将结果写入 Redis 缓存,便于后续访问。

代码示例(使用 Python + Redis):

import redis
import time# 连接 Redis
cache = redis.Redis(host='localhost', port=6379, decode_responses=True)def get_data_from_db(key):""" 模拟数据库查询 """time.sleep(1)  # 模拟查询延迟return f"Value of {key}"def get_data(key):""" 先查 Redis,未命中则查数据库,并存入 Redis """value = cache.get(key)if value is None:print("Cache Miss, Fetching from DB...")value = get_data_from_db(key)cache.setex(key, 3600, value)  # 设置 1 小时过期else:print("Cache Hit!")return value# 测试
print(get_data("user:1001"))
print(get_data("user:1001"))

优点:

  • 适用于 读多写少 的场景,如热点数据查询。
  • 缓存有效期 可控制,避免长期存储过期数据。

缺点:

  • 可能会遇到 缓存穿透缓存击穿缓存雪崩 等问题(后面会详细讲解)。

2.2. 写穿透缓存(Write-through Cache)

原理:

  1. 写数据时,同时更新数据库和 Redis,保证数据一致性;
  2. 读取数据时,先查 Redis,命中直接返回,未命中则从数据库查询,并更新缓存

代码示例:

def update_data(key, value):""" 更新数据库,同时更新缓存 """print("Updating database...")# 这里模拟更新数据库time.sleep(1)  # 模拟写入延迟cache.setex(key, 3600, value)  # 立即更新缓存print("Cache updated!")# 测试
update_data("user:1001", "Updated Value")
print(get_data("user:1001"))  # 应该返回新的值

优点:

  • 适用于 读写频率相近 的场景,比如电商库存、用户账户余额。
  • 由于写时更新缓存,能够 减少缓存击穿问题

缺点:

  • 每次写操作都要更新缓存,可能会导致 写压力增加

3. 解决缓存常见问题

3.1. 缓存穿透

问题:

  • 用户请求的数据在数据库不存在,导致每次请求都 无法命中缓存,直接查询数据库
  • 可能导致数据库 压力剧增,甚至崩溃。

解决方案:

  1. 缓存空值:对于查询结果为空的 key,也存入 Redis,避免频繁查询数据库

    value = cache.get("user:9999")
    if value is None:db_value = get_data_from_db("user:9999")if db_value is None:cache.setex("user:9999", 3600, "NULL")  # 存一个空值else:cache.setex("user:9999", 3600, db_value)
    
  2. 布隆过滤器(Bloom Filter):在请求 Redis 之前,先用布隆过滤器判断 key 是否可能存在。


3.2. 缓存击穿

问题:

  • 某个热点 key 过期 后,大量并发请求同时查询数据库,造成数据库压力过大。

解决方案:

  1. 设置合理的过期时间,采用 随机过期时间 避免多个 key 同时过期。

  2. 互斥锁:在缓存失效后,只有 一个线程更新缓存,其他线程等待:

    lock = cache.setnx("lock:user:1001", 1)  # 尝试加锁
    if lock:value = get_data_from_db("user:1001")cache.setex("user:1001", 3600, value)  # 更新缓存cache.delete("lock:user:1001")  # 释放锁
    

3.3. 缓存雪崩

问题:

  • 大量缓存 key 同时过期,导致大量请求直接访问数据库,造成宕机风险。

解决方案:

  1. 缓存 key 设定不同的过期时间(如 3600 + random(600) 秒)。
  2. 使用 Redis 集群,分散缓存压力。
  3. 预加载数据,定期更新缓存,避免大规模过期。

4. Redis 高级优化技巧

4.1. 使用合适的数据结构

  • 字符串(String):适用于简单的 key-value 存储,如用户信息缓存
  • 哈希(Hash):适用于存储结构化数据。
  • 列表(List):适用于消息队列。
  • 集合(Set):适用于去重操作。
  • 有序集合(Sorted Set):适用于排行榜。

4.2. Redis LRU 淘汰策略

CONFIG SET maxmemory-policy allkeys-lru

4.3. 采用 Redis 分布式架构

  • 主从复制:适用于读多写少的场景。
  • Redis 哨兵:提供自动故障恢复。
  • Redis Cluster:支持 分片存储

总结

Redis 作为高效缓存,能够极大提高数据访问速度,降低数据库压力。但在实际使用中,需要结合缓存策略、淘汰策略和分布式架构,避免缓存穿透、击穿和雪崩等问题,实现高可用、高性能的缓存系统。


http://www.ppmy.cn/devtools/152289.html

相关文章

安装matlab2024a错误license checkout failed Error-8

问题: 忘记截图了,借用博主的图片。 记得安装过程中,目标网址才是你的安装地址,而不是前面的安装包地址。 解决方法: 1.将破解文件中"Crack\R2020a\bin\win64\matlab_startup_plugins\lmgrimpl"目录下的l…

【设计模式-行为型】策略模式

一、什么是策略模式 要说策略模式,那么我脑袋里有一个例子,是一个策略模式使用的大家,那就是在周星驰的电影《国产凌凌漆》中的----达文西。 达文西:“没错就是我,请叫我文西”。 电影中,角色达文西以其机智…

matlab计算功率谱的四种方法

%{ 计算功率谱的四种方法,前三种方法采用归一化频率,第四种方法采用原始频率 %} clear;clc;close all; % 生成一个示例时间序列(例如正弦波加上一些随机噪声) t 0:0.01:100-0.1; % 时间向量 % x sin(t) 0.5 * randn(size(t))…

【opencv】第10章 角点检测

第10章 角点检测 10.1 Harris角点检测 10.1.1 兴趣点与角点 在图像处理和与计算机视觉领域,兴趣点(interest points),也被称作关键点 (key points)、特 征 点(feature points)。它被大量用于解决物体识别、图像识别、 图像匹配、视觉跟踪、三维重建等一系列的问题…

GoLang 微服务学习笔记

https://www.bilibili.com/video/BV1Gg4y1u77D/?spm_id_from333.337.search-card.all.click&vd_source707ec8983cc32e6e065d5496a7f79ee6 目录 第1讲:为什么说云原生重构了互联网产品开发模式? https://www.bilibili.com/video/BV1Gg4y1u77D?spm_i…

MySQL字符串函数详解

简介 本文主要讲解MySQL中的字符串函数,包括:left、right、ltrim、rtrim、trim、upper、lower、lpad、rpad、concat、concat_ws、instr、substr、length、char_length、replace、reverse、char、ascii。 left函数 语法:left(str,length)&a…

Kotlin 2.1.0 入门教程(三)

变量 在 Kotlin 中,可以通过关键字 val 或 var 声明变量,后跟变量名称。 使用 val 关键字声明只能赋值一次的变量。这些是不可变的、只读的局部变量,初始化后不能重新赋值。 fun main() {val a 1val b: Int 2println("a $a, b $b…

【STM32-学习笔记-8-】I2C通信

文章目录 I2C通信Ⅰ、硬件电路Ⅱ、IIC时序基本单元① 起始条件② 终止条件③ 发送一个字节④ 接收一个字节⑤ 发送应答⑥ 接收应答 Ⅲ、IIC时序① 指定地址写② 当前地址读③ 指定地址读 Ⅳ、MPU6050---6轴姿态传感器(软件I2C)1、模块内部电路2、寄存器地…