使用redis的5种常用场景

news2025/1/9 7:26:14

文章目录

  • 1. 缓存热点数据
  • 2. 分布式锁
  • 3. 计数器和限流器
  • 4. 消息队列
  • 5. 会话管理
  • 总结

在日常开发工作中,Redis作为一款高性能的内存数据库,凭借其强大的功能特性和卓越的性能表现,已经成为了许多项目中不可或缺的组件。本文将详细介绍Redis在实际工作中最常见的5种应用场景,并附上具体的代码实现。

1. 缓存热点数据

在高并发系统中,大量的数据库查询会对系统性能造成严重影响。通过Redis缓存热点数据,可以显著减少数据库的访问压力,提升系统响应速度。

实现示例:

public class ProductService {
    @Autowired
    private RedisTemplate<String, Product> redisTemplate;
    @Autowired
    private ProductMapper productMapper;
    
    private static final String PRODUCT_KEY_PREFIX = "product:";
    private static final long CACHE_TIMEOUT = 3600; // 缓存过期时间:1小时
    
    public Product getProduct(Long productId) {
        // 构建Redis键
        String redisKey = PRODUCT_KEY_PREFIX + productId;
        
        // 首先从Redis中查询
        Product product = redisTemplate.opsForValue().get(redisKey);
        
        if (product != null) {
            // 缓存命中,直接返回
            return product;
        }
        
        // 缓存未命中,查询数据库
        product = productMapper.selectById(productId);
        
        if (product != null) {
            // 将查询结果存入Redis
            redisTemplate.opsForValue().set(redisKey, product, CACHE_TIMEOUT, TimeUnit.SECONDS);
        }
        
        return product;
    }
}

2. 分布式锁

在分布式系统中,常常需要控制对共享资源的访问。Redis的SETNX命令提供了一种实现分布式锁的简单方法。

实现示例:

public class RedisLock {
    @Autowired
    private StringRedisTemplate redisTemplate;
    
    private static final long LOCK_TIMEOUT = 10000; // 锁超时时间:10秒
    
    public boolean acquireLock(String lockKey, String requestId) {
        // 使用SETNX命令尝试获取锁
        Boolean isLocked = redisTemplate.opsForValue().setIfAbsent(
            lockKey, 
            requestId,
            LOCK_TIMEOUT,
            TimeUnit.MILLISECONDS
        );
        
        return Boolean.TRUE.equals(isLocked);
    }
    
    public boolean releaseLock(String lockKey, String requestId) {
        // 使用Lua脚本确保原子性操作
        String script = "if redis.call('get', KEYS[1]) == ARGV[1] then "
                     + "return redis.call('del', KEYS[1]) "
                     + "else "
                     + "return 0 "
                     + "end";
                     
        Long result = redisTemplate.execute(
            new DefaultRedisScript<>(script, Long.class),
            Collections.singletonList(lockKey),
            requestId
        );
        
        return Long.valueOf(1).equals(result);
    }
}

3. 计数器和限流器

Redis的INCR命令可以原子性地实现计数器功能,非常适合用于实现访问统计和接口限流。

实现示例:

public class RateLimiter {
    @Autowired
    private StringRedisTemplate redisTemplate;
    
    private static final String RATE_LIMIT_PREFIX = "rate:limit:";
    private static final int MAX_REQUESTS = 100; // 每分钟最大请求次数
    
    public boolean isAllowed(String userId) {
        String key = RATE_LIMIT_PREFIX + userId;
        
        // 获取当前时间的分钟数作为窗口
        long currentMinute = System.currentTimeMillis() / (60 * 1000);
        String windowKey = key + ":" + currentMinute;
        
        // 原子性增加计数
        Long count = redisTemplate.opsForValue().increment(windowKey);
        
        if (count == 1) {
            // 设置过期时间为1分钟
            redisTemplate.expire(windowKey, 60, TimeUnit.SECONDS);
        }
        
        // 判断是否超过限制
        return count <= MAX_REQUESTS;
    }
}

4. 消息队列

虽然Redis不是专门的消息队列系统,但它的List数据结构配合LPUSH和BRPOP命令可以实现简单的消息队列功能。

实现示例:

public class RedisMessageQueue {
    @Autowired
    private StringRedisTemplate redisTemplate;
    
    private static final String QUEUE_KEY = "message:queue";
    
    // 生产者
    public void sendMessage(String message) {
        redisTemplate.opsForList().leftPush(QUEUE_KEY, message);
    }
    
    // 消费者
    @Async
    public void consumeMessage() {
        while (true) {
            try {
                // 阻塞式获取消息
                List<String> message = redisTemplate.opsForList().rightPop(
                    QUEUE_KEY,
                    30,
                    TimeUnit.SECONDS
                );
                
                if (message != null) {
                    // 处理消息
                    processMessage(message);
                }
            } catch (Exception e) {
                log.error("处理消息出错", e);
            }
        }
    }
    
    private void processMessage(List<String> message) {
        // 具体的消息处理逻辑
    }
}

5. 会话管理

Redis的Hash数据结构非常适合存储用户会话信息,支持部分字段的更新,且可以设置过期时间。

实现示例:

public class SessionManager {
    @Autowired
    private StringRedisTemplate redisTemplate;
    
    private static final String SESSION_KEY_PREFIX = "session:";
    private static final long SESSION_TIMEOUT = 1800; // 会话超时时间:30分钟
    
    public void saveSession(String sessionId, Map<String, String> sessionData) {
        String key = SESSION_KEY_PREFIX + sessionId;
        
        // 使用Hash结构存储会话数据
        redisTemplate.opsForHash().putAll(key, sessionData);
        
        // 设置过期时间
        redisTemplate.expire(key, SESSION_TIMEOUT, TimeUnit.SECONDS);
    }
    
    public void updateSessionField(String sessionId, String field, String value) {
        String key = SESSION_KEY_PREFIX + sessionId;
        
        // 更新单个字段
        redisTemplate.opsForHash().put(key, field, value);
        
        // 刷新过期时间
        redisTemplate.expire(key, SESSION_TIMEOUT, TimeUnit.SECONDS);
    }
    
    public Map<Object, Object> getSession(String sessionId) {
        String key = SESSION_KEY_PREFIX + sessionId;
        return redisTemplate.opsForHash().entries(key);
    }
    
    public void deleteSession(String sessionId) {
        String key = SESSION_KEY_PREFIX + sessionId;
        redisTemplate.delete(key);
    }
}

总结

Redis凭借其出色的性能和丰富的数据结构,在实际工作中可以解决很多具体问题。上述5种场景只是Redis应用的冰山一角,在实际开发中,我们还可以根据具体需求,结合Redis的特性来设计更多的解决方案。

在使用Redis时,需要注意以下几点:

  1. 合理设置过期时间,避免内存占用过大
  2. 注意缓存与数据库的一致性问题
  3. 在分布式环境下要考虑并发问题
  4. 根据实际需求选择合适的数据结构
  5. 定期监控Redis的内存使用情况和性能指标

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2273654.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

vue数据请求通用方案:axios的options都有哪些值

Axios 是一个基于 promise 的 HTTP 库&#xff0c;可以用在浏览器和 Node.js 中。 在使用 Axios 发送请求时&#xff0c;可以通过传递一个配置对象来指定请求的各种选项。 以下是一些常用的 Axios 配置选项及其说明&#xff1a; 1.url: &#xff08;必需&#xff09;请求的 …

阿里mod_asr3.0集成webrtc静音算法

alibabacloud-nls-cpp-sdk-master 先到阿里官网下载nls库的源代码&#xff0c;编译生成对应的库文件和头文件。 我编译的放到了以下目录。 /home/jp/2025/alibabacloud-nls-cpp-sdk-master/build/install/NlsSdk3.X_LINUX/include/ /home/jp/2025/alibabacloud-nls-cpp-sdk-…

【大模型】百度千帆大模型对接LangChain使用详解

目录 一、前言 二、LangChain架构与核心组件 2.1 LangChain 核心架构 2.2 LangChain 核心组件 三、环境准备 3.1 前置准备 3.1.1 创建应用并获取apikey 3.1.2 开通付费功能 3.2 获取LangChain文档 3.3 安装LangChain依赖包 四、百度千帆大模型对接 LangChain 4.1 LL…

【51单片机零基础-chapter6:LCD1602调试工具】

实验0-用显示屏LCD验证自己的猜想 如同c的cout,前端的console.log() #include <REGX52.H> #include <INTRINS.H> #include "LCD1602.h" int var0; void main() {LCD_Init();LCD_ShowNum(1,1,var211,5);while(1){;} }实验1-编写LCD1602液晶显示屏驱动函…

Mysql--基础篇--事务(ACID特征及实现原理,事务管理模式,隔离级别,并发问题,锁机制,行级锁,表级锁,意向锁,共享锁,排他锁,死锁,MVCC)

在MySQL中&#xff0c;事务&#xff08;Transaction&#xff09;是一组SQL语句的集合&#xff0c;这些语句一起被视为一个单一的工作单元。事务具有ACID特性&#xff0c;确保数据的一致性和完整性。通过事务&#xff0c;可以保证多个操作要么全部成功执行&#xff0c;要么全部不…

使用VUE3创建个人静态主页

使用VUE3创建个人静态主页 &#x1f31f; 前言&#x1f60e;体验&#x1f528; 具体实现✨ 核心功能&#x1f3d7;️ 项目结构&#x1f680; 用这个项目部署 Git Page &#x1f4d6; 参考 &#x1f31f; 前言 作为开发者或者内容创作者&#xff0c;我们经常需要创建静态网页&a…

鸿蒙APP之从开发到发布的一点心得

引言&#xff1a; 做鸿蒙开发大概有1年左右时间了&#xff0c;从最开始的看官方文档、看B站视频&#xff0c;到后来成功发布两款个人APP&#xff08;房贷计算极简版、时简时钟 轻喷&#xff0c;谢谢&#xff09;。简单描述一下里边遇到的坑以及一些经历吧。 学习鸿蒙开发 个…

【HTML+CSS+JS+VUE】web前端教程-3-标题标签

标题介绍与应用 标题是通过<h1>-<h6>标签进行定义的 <h1>定义最大的标题 <h6>定义最小的标题<h1>一级标题</h1> <h2>二级标题</h2> <h3>三级标题</h3> <h4>四级标题</h4> <h5>五级标题</h5…

Web应用安全-漏洞扫描器设计与实现

摘 要 随着Web2.0、社交网络、微博等一系列新型的互联网产品的诞生&#xff0c;基于Web环境的互联网应用越来越广泛&#xff0c;企业信息化的过程中各种应用都架设在Web平台上。Web应用的迅速发展也引起黑客们的强烈关注&#xff0c;接踵而至的就是Web安全威胁的凸显&#xff…

SpringBootWeb案例-1(day10)

准备工作 需求 & 环境搭建 需求说明 环境搭建 步骤&#xff1a; 准备数据库表(dept、emp)创建 springboot 工程&#xff0c;引入对应的起步依赖&#xff08;web、mybatis、mysql 驱动、lombok&#xff09;配置文件 application.properties 中引入 mybatis 的配置信息&…

《Spring Framework实战》5:Spring Framework 概述

欢迎观看《Spring Framework实战》视频教程 Spring 使创建 Java 企业应用程序变得容易。它为您提供一切 需要在企业环境中采用 Java 语言&#xff0c;并支持 Groovy 和 Kotlin 作为 JVM 上的替代语言&#xff0c;并且可以灵活地创建许多 类型的架构。从 Spring Framework 6.0 开…

逆向百例3——全国招标公告

所有内容仅供学习交流使用&#xff01; 全国招标公告搜索引擎 解决开发者工具打不开问题接口分析解密数据 本次逆向主要是解决该页面接口返回的数据&#xff0c;但是接口所返回的数据是密文&#xff0c;经过分析用的是标准DES&#xff0c;解密成明文即可。 解决开发者工具打不…

windows10下安装Microsoft SQL Server 2016

一、下载安装包 网站&#xff1a;MSDN, 我告诉你 - 做一个安静的工具站 选择需要的版本&#xff0c;点击详细信息&#xff0c;复制ed2k链接&#xff0c;打开eMule或迅雷&#xff0c;新建下载&#xff0c;粘贴链接&#xff0c;开始下载。 下载好的文件是一个.iso镜像文件。 二、…

一、二极管(应用篇)

1.5普通二极管应用 1.5.1钳位电路 利用二极管的固定的导通电压&#xff0c;在二极管处并联用电器&#xff0c;达到用电器的工作电压相对稳定。如果电源处有尖峰电压&#xff0c;则可以通过二极管导入到5v的电源内&#xff0c;防止此尖峰电压干扰用电器 &#xff0c;起到对电路的…

SpringCloud系列教程:微服务的未来(十)服务调用、注册中心原理、Nacos注册中心

本博客将重点介绍服务调用和注册中心的原理&#xff0c;特别是以 Nacos 为例&#xff0c;详细讲解 Nacos 注册中心如何实现服务的注册与发现。同时&#xff0c;分析 Nacos 注册中心在分布式微服务中的应用&#xff0c;帮助开发者更好地理解其工作机制。 目录 前言 微服务拆分…

eNSP之家----ACL实验入门实例详解(Access Control List访问控制列表)(重要重要重要的事说三遍)

ACL实验&#xff08;Access Control List访问控制列表&#xff09;是一种基于包过滤的访问控制技术&#xff0c;它可以根据设定的条件对接口上的数据包进行过滤&#xff0c;允许其通过或丢弃。访问控制列表被广泛地应用于路由器和三层交换机。 准备工作 在eNSP里面部署设备&a…

benchANT 性能榜单技术解读 Part 1:写入吞吐

近期&#xff0c;国际权威数据库性能测试榜单 benchANT 更新了 Time Series: Devops&#xff08;时序数据库&#xff09;场景排名&#xff0c;KaiwuDB 数据库在 xsmall 和 small 两类规格下的时序数据写入吞吐、查询吞吐、查询延迟、成本效益等多项指标刷新榜单原有数据纪录。在…

消息队列MQ(二)

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 MQ学习笔记 前言一、发送者的可靠性1. 生产者重试机制2. 生产者确认机制3. 实现生产者确认 二、MQ的可靠性1. 数据持久化2. LazyQueue 前言 在用MQ实现异步调用时&#xff0…

HTML基础入门——简单网页页面

目录 一&#xff0c;网上转账电子账单 ​编辑 1&#xff0c;所利用到的标签 2&#xff0c;代码编写 3&#xff0c;运行结果 二&#xff0c;李白诗词 1&#xff0c;所用到的标签 2&#xff0c;照片的编辑 3&#xff0c;代码编写 4&#xff0c;运行结果 一&#xff0c;网…

365天深度学习训练营:第N2周:构建词典

&#x1f368; 本文为&#x1f517;365天深度学习训练营中的学习记录博客&#x1f356; 原作者&#xff1a;K同学啊 本周任务: 使用N1周的.txt 文件构建词典&#xff0c;停用词请自定义 1. 导入数据 from torchtext.vocab import build_vocab_from_iterator from collection…