当前位置: 首页 > news >正文

京东二面:Redis的使用场景有哪些?别说你只用过缓存

前言

最近面试季,好多小伙伴都反馈同一个问题:“面试官问我Redis的使用场景,我只说了缓存,结果当场被怼——‘就这?’”

其实,Redis远不止“缓存”这么简单。

它是一个多才多艺的选手,凭借内存读写、丰富的数据结构、原子操作、持久化、高可用等特性,几乎渗透到了后端开发的所有角落。

今天这篇文章就专门跟大家一起聊聊Redis的11种使用场景,希望对你会有所帮助。

更多项目实战在Java突击队网:susan.net.cn/project

场景一:高速缓存(最经典)

业务场景

查询商品详情、用户信息等高频读数据,把数据库扛不住的读压力放到Redis。

代码示例(旁路缓存模式)

@Service
public class ProductService {@Autowiredprivate RedisTemplate<String, Product> redisTemplate;@Autowiredprivate ProductMapper productMapper;public Product getProduct(Long id) {String cacheKey = "product:" + id;// 1. 读缓存Product product = redisTemplate.opsForValue().get(cacheKey);if (product != null) {return product;}// 2. 缓存未命中,查DBproduct = productMapper.selectById(id);if (product != null) {// 3. 写缓存,随机过期时间防雪崩int expire = 300 + new Random().nextInt(60);redisTemplate.opsForValue().set(cacheKey, product, expire, TimeUnit.SECONDS);}return product;}
}

优点:性能从几十毫秒降到亚毫秒级,能扛极高并发。
缺点:存在短时间数据不一致(最终一致性)。
适用场景:商品详情、配置信息、用户会话等读多写少的数据。

场景二:分布式锁

业务场景

秒杀扣库存、订单防重、定时任务防重复执行——多进程争抢同一资源时,需要一把全局锁。

推荐方案:Redisson(自动续期、可重入)

@Service
public class SeckillService {@Autowiredprivate RedissonClient redissonClient;public void doSeckill(Long userId, Long productId) {String lockKey = "lock:seckill:" + productId;RLock lock = redissonClient.getLock(lockKey);try {// 尝试加锁,最多等待3秒,锁默认30秒(看门狗自动续期)if (lock.tryLock(3, TimeUnit.SECONDS)) {// 扣库存逻辑...int stock = getStock(productId);if (stock > 0) {updateStock(productId, stock - 1);} else {throw new RuntimeException("已售罄");}} else {throw new RuntimeException("系统繁忙,请稍后再试");}} catch (InterruptedException e) {Thread.currentThread().interrupt();} finally {if (lock.isHeldByCurrentThread()) {lock.unlock();}}}
}

⚠️ 新手坑:直接用SET NX EX,容易忘了续期或释放锁。强烈建议用Redisson。

优点:自动续期,可重入,支持主从/哨兵/集群。
缺点:引入额外客户端,比原生命令重一点。
适用场景:任何需要互斥的分布式操作。

场景三:计数器 & 限流器

业务场景

统计页面PV、接口调用次数、短信发送频率控制。

固定窗口限流(最简单的实现)

public boolean isAllowed(String userId) {String key = "rate:sendSms:" + userId;Long count = redisTemplate.opsForValue().increment(key);if (count == 1) {redisTemplate.expire(key, 60, TimeUnit.SECONDS);}return count <= 5;  // 每分钟最多5次
}

优点:原子操作,性能极高。
缺点:固定窗口存在临界突刺(边界处可能两倍流量)。更严格的可用滑动窗口(ZSET实现)。
适用场景:API限流、短信验证码、登录错误次数。

场景四:排行榜 / 有序集合

业务场景

游戏积分榜、热搜榜单、销量排行。

使用Sorted Set

// 添加玩家分数
redisTemplate.opsForZSet().add("rank:game:202605", "player:1001", 9980);
redisTemplate.opsForZSet().add("rank:game:202605", "player:1002", 10200);// 获取前三名(降序)
Set<ZSetOperations.TypedTuple<String>> top3 =redisTemplate.opsForZSet().reverseRangeWithScores("rank:game:202605", 0, 2);

优点:自动排序,查询极快(O(logN))。
缺点:内存占用随数据量线性增长。
适用场景:实时排行榜、热点文章、商品热销榜。

场景五:最新动态 / 时间线

业务场景

微博时间线、Feed流、最新评论。

使用List(左侧推入,右侧截取)

// 发表一条新动态
String feedKey = "feeds:user:" + userId;
redisTemplate.opsForList().leftPush(feedKey, JSON.toJSONString(post));
// 只保留最近100条
redisTemplate.opsForList().trim(feedKey, 0, 99);

优点:写入快,分页方便。
缺点:无法做复杂排序,且只能按时间倒序。
适用场景:简单的“最新N条”列表,如用户动态、通知。

场景六:社交关系(集合运算)

业务场景

共同关注、好友推荐、可能认识的人。

// 用户A关注的人
redisTemplate.opsForSet().add("follow:1001", "1002", "1003", "1004");
// 用户B关注的人
redisTemplate.opsForSet().add("follow:1002", "1003", "1005");// 共同关注
Set<String> intersect = redisTemplate.opsForSet().intersect("follow:1001", "follow:1002");
// 推荐关注(B关注但A未关注)
Set<String> diff = redisTemplate.opsForSet().difference("follow:1002", "follow:1001");

优点:集合运算毫秒级,代码简洁。
缺点:大key(比如千万粉丝)会导致慢查询。
适用场景:社交关系、权限标签、黑白名单。

场景七:轻量级消息队列(Stream / List)

业务场景

异步处理、削峰填谷、日志收集。

使用Redis 5.0+ Stream(支持消费组)

// 生产者
Map<String, String> msg = new HashMap<>();
msg.put("orderId", "ORD-123");
redisTemplate.opsForStream().add("order:stream", msg);// 消费者
List<MapRecord<String, Object, Object>> records =redisTemplate.opsForStream().read(Consumer.from("order-group", "c1"),StreamReadOptions.empty().count(10),StreamOffset.create("order:stream", ReadOffset.lastConsumed()));

优点:无需额外组件,支持ACK、消费者组。
缺点:持久化可靠性不如RocketMQ/Kafka。
适用场景:内部任务分发、异步通知、不要求严格有序的场景。

场景八:布隆过滤器(防缓存穿透)

业务场景

拦截恶意请求,过滤肯定不存在的数据,避免打到数据库。

使用Redisson布隆过滤器

RBloomFilter<String> bloom = redissonClient.getBloomFilter("filter:productId");
bloom.tryInit(1000000L, 0.01); // 容量100万,误判率1%
// 初始化时把合法ID加入
for (Long id : allProductIds) {bloom.add(id.toString());
}// 查询前先过滤
if (!bloom.contains(productId.toString())) {return null;  // 直接返回空,不查DB
}

优点:内存占用极小(1亿条仅约120MB)。
缺点:有误判率(宁可错杀一千,不放过一个)。
适用场景:防止恶意穿透、爬虫IP去重、黑名单过滤。

场景九:轻量级NoSQL(Hash存储对象)

业务场景

存储动态属性、用户Session、配置信息。

// 存储用户信息
Map<String, String> user = new HashMap<>();
user.put("name", "张三");
user.put("age", "28");
redisTemplate.opsForHash().putAll("user:1001", user);// 获取单个字段
String name = (String) redisTemplate.opsForHash().get("user:1001", "name");
// 修改单个字段
redisTemplate.opsForHash().put("user:1001", "age", "29");

优点:比JSON序列化整个对象更节省内存,支持部分更新。
缺点:hash结构不适合深层次嵌套。
适用场景:用户资料、购物车(field为商品ID,value为数量)。

场景十:数据去重 / 唯一性判断

业务场景

UV统计、不能重复参加的抽奖、防重复提交。

使用Set或HyperLogLog

// 方式1:Set(精确但内存大)
Boolean success = redisTemplate.opsForSet().add("draw:202605", userId);
if (Boolean.TRUE.equals(success)) {// 第一次参加
}// 方式2:HyperLogLog(内存极小,有误差)
redisTemplate.opsForHyperLogLog().add("uv:20260501", userId);
Long uv = redisTemplate.opsForHyperLogLog().size("uv:20260501");

优点:Set保证绝对精确;HyperLogLog内存极省(12KB)。
缺点:HyperLogLog有0.81%误差,不适合精确计数。
适用场景:抽奖去重、日活统计、爬虫URL去重。

场景十一:延时队列(使用Sorted Set)

业务场景

订单30分钟未支付自动取消、延迟通知。

// 添加延时任务(score为执行时间戳)
long execTime = System.currentTimeMillis() + 30 * 60 * 1000;
redisTemplate.opsForZSet().add("delay:order", orderId, execTime);// 消费者定时扫描
Set<String> tasks = redisTemplate.opsForZSet().rangeByScore("delay:order", 0, System.currentTimeMillis());
for (String orderId : tasks) {// 处理超时订单if (redisTemplate.opsForZSet().remove("delay:order", orderId) > 0) {cancelOrder(orderId);}
}

优点:精准到毫秒,实现简单。
缺点:需要定时轮询,数据量大时性能下降。
适用场景:订单超时、定时提醒、会话过期清理。

更多项目实战在Java突击队网:susan.net.cn/project

总结

Redis不是只能做缓存。

当你遇到以下问题时,可以优先想想Redis:

  • 快不快? → 缓存、计数器
  • 要不要排队? → 消息队列
  • 要不要去重? → 集合、布隆过滤器
  • 要不要按分数排名? → Sorted Set
  • 多机器抢资源? → 分布式锁

下面是一张速查表,方便你回忆:

场景 数据结构 核心命令
缓存 String GET/SET
分布式锁 String + Redisson SET NX / tryLock
计数器 String INCR
排行榜 ZSET ZADD, ZREVRANGE
最新动态 List LPUSH, LTRIM
共同关注 Set SINTER, SDIFF
消息队列 Stream XADD, XREADGROUP
布隆过滤 -> Redisson RBloomFilter
对象存储 Hash HMSET, HGET
UV去重 HyperLogLog PFADD, PFCOUNT
延时队列 ZSET ZADD, ZRANGEBYSCORE

最后,送大家一句话:Redis的数据结构就是你的武器库,用对结构,往往能事半功倍。

希望这11个场景能帮你打开思路,下次面试时遇到“Redis能做什么”的问题,你可以自信地展开聊上十分钟。

http://www.jsqmd.com/news/769312/

相关文章:

  • Flutter 字体生效原理解析
  • 品牌擦片机制造商哪家好 - mypinpai
  • 喜茶代金券回收哪里好 怎么操作更省心 - 畅回收小程序
  • 企业如何利用多模型聚合平台优化AI应用开发成本与效率
  • 别再被web-view盖住了!用uniapp的cover-view给小程序网页加个授权弹窗(附完整代码)
  • 你的Python脚本吃掉了多少内存?用psutil进行程序性能分析与资源泄漏排查实战
  • 解决方案:PvZ Toolkit如何通过内存注入技术重塑植物大战僵尸的游戏体验?
  • 深入ZynqMP启动流程:从BootROM到Linux桌面,一张图看懂Petalinux每个文件的作用
  • 2026年贵阳装修公司排名完全指南:从预算透明到品质交付的深度横评 - 年度推荐企业名录
  • Step3.5 Flash 大模型技术深度解析:稀疏 MoE、混合注意力与 MTP 的高效推理革命
  • 选购酒店床上用品,哪个品牌好? - mypinpai
  • 2026年贵阳装修公司排名|闭口合同+VR设计+环保承诺的靠谱整装公司怎么选 - 年度推荐企业名录
  • 【微波辐射】基于matlab模拟综合孔径微波辐射成像仿真,含校正前后傅氏反演图像 Y阵型反演图像
  • TensorFlow模型快速部署:基于Gradio的AI演示界面构建指南
  • 免费解锁电脑性能的完整指南:Universal x86 Tuning Utility终极教程
  • 卡梅德生物技术快报:微生物基因敲入工程化构建甘露醇高产菌株
  • 2026年郫都区西装定制哪家靠谱?琪诺服装口碑佳 - mypinpai
  • 年省超200万!除垢剂实战案例深度解析 - 速递信息
  • **零基础小白用 GitHub 和 OpenCode 写代码入门教程(超详细保姆级)**
  • SARAH技术解析:实时自适应动作生成的突破与应用
  • 告别复制粘贴!用STM32CubeMX HAL库驱动ESP8266的保姆级避坑指南
  • 玫瑰痤疮可用防晒霜推荐来了,这4款温和防晒修护力拉满 - 全网最美
  • 远程命令执行系统架构设计:从Agent模型到gRPC安全通信实践
  • MakeFile编译管理工具
  • Go function - 有关function我能告诉你的一切
  • 神经网络参数化缩放(µP)原理与实践指南
  • Claude Code 免费使用指南:free-claude-code 代理方案全解析(2026)
  • 2026年贵阳装修公司排名指南:预算透明+环保可信的五大靠谱品牌深度横评 - 年度推荐企业名录
  • 模型量化鲁棒性优化:学习率调度与权重平均技术
  • dnSpy配置管理实战:从个人工作流到团队协作的进阶指南