Redis

关注公众号 jb51net

关闭
首页 > 数据库 > Redis > Redis 未读消息计数

Redis实现未读消息计数的示例代码

作者:她说..

本文介绍了合伙人系统中产品分配流程及其未读计数的实现方案,通过Redis工具类,实现了未读计数的新增、查询和重置功能,感兴趣的可以了解一下

一、业务场景说明

在合伙人系统的产品分配流程中,存在两种分配模式,核心差异在于是否需要审核:

为提升城市合伙人的操作效率,小程序需在 “分配产品页面” 为城市合伙人显示待审核数,实时提醒其待处理的间接分配申请,而未读计数的存储与管理是实现该功能的核心。

二、未读计数的实现方案

1. 核心触发逻辑

在 “间接分配审核接口” 的最后,通过一行代码触发未读计数更新,直接调用工具类完成城市合伙人未读数量的累加:

// 添加未读数(默认新增1条待审核提醒)
appletRedisUtil.addUnreadCount(cityPartner.getId());

2. 核心工具类:AppletRedisUtil

工具类基于 Redis 实现未读计数的 “新增、查询、重置” 全流程管理,代码与逻辑解析如下:

import jakarta.annotation.Resource;
import org.springblade.business.constant.RedisKeyConstant;
import org.springblade.business.pojo.entity.ProductApplyRecord;
import org.springblade.business.service.ProductApplyRecordService;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.stereotype.Component;
import java.util.Objects;
 
@Component // 注入Spring容器,全局可用
public class AppletRedisUtil {
 
    @Resource
    private RedisTemplate<String, Long> redisTemplate; // 操作Redis的核心组件
 
    // 1. 重载方法:默认给城市合伙人新增1条未读消息
    public void addUnreadCount(Long miniUserId) {
        addUnreadCount(miniUserId, 1L);
    }
 
    // 2. 核心方法:支持自定义新增未读条数,含数据准确性校验
    public void addUnreadCount(Long miniUserId, Long value) {
        // 2.1 工具类无法直接注入Service,通过Spring上下文获取
        ProductApplyRecordService productApplyRecordService = SpringContextUtil.getBean(ProductApplyRecordService.class);
        
        // 2.2 查该城市合伙人的总申请数(未读上限,避免未读数超过实际总数)
        Long totalCount = productApplyRecordService.lambdaQuery()
            .eq(ProductApplyRecord::getCityPartnerId, miniUserId)
            .count();
        
        // 2.3 查当前Redis中的未读数量(空值兜底返回0,避免空指针)
        Long currentUnread = getUnreadCount(miniUserId);
        
        // 2.4 修正未读数量:若累加后超总申请数,取总申请数(防止数据异常)
        value = (currentUnread + value) > totalCount ? totalCount : (currentUnread + value);
        
        // 2.5 更新Redis:用“常量前缀+用户ID”作为key,原子自增未读数量
        redisTemplate.opsForValue().increment(RedisKeyConstant.PRODUCT_APPLY_UNREAD_NUM + miniUserId, value);
    }
 
    // 3. 查询未读数量:空值兜底,确保返回非null
    public Long getUnreadCount(Long miniUserId) {
        String key = RedisKeyConstant.PRODUCT_APPLY_UNREAD_NUM + miniUserId;
        Long unread = redisTemplate.opsForValue().get(key);
        return Objects.isNull(unread) ? 0L : unread;
    }
 
    // 4. 重置未读数量:先置0再删key,确保状态彻底清空
    public void resetUnreadCount(Long miniUserId) {
        String key = RedisKeyConstant.PRODUCT_APPLY_UNREAD_NUM + miniUserId;
        redisTemplate.opsForValue().set(key, 0L);
        redisTemplate.delete(key);
    }
}

工具类核心特点

三、选型:为什么用 Redis 而非 MySQL?

未读计数的核心诉求是 “快、并发安全、简单”,Redis 完美匹配这些需求,而 MySQL 更擅长 “复杂查询、事务一致性、永久存储”,具体优势对比如下:

1. 性能碾压:高频场景响应速度差 3 个量级

特性

Redis(内存数据库)

MySQL(磁盘数据库)

响应时间

微秒级(1μs = 10⁻⁶秒)

毫秒级(1ms = 10⁻³ 秒)

每秒读写能力

数万~数十万次

千级次

高频场景表现

无卡顿,轻松扛住并发(如同时 100 个申请提交)

易出现 “查询卡顿”“写入排队”,拖慢数据库

2. 操作更轻量:避免 MySQL 复杂开销

3. 天然并发安全:解决 MySQL 更新冲突

当多个请求同时修改同一城市合伙人的未读计数时(如同时 2 条申请提交):

4. 缓存特性适配:减少数据库压力

5. 灵活扩展:支持更多交互场景

Redis 的特性可轻松满足未来扩展需求,MySQL 难以实现:

四、注意事项:Redis 数据安全兜底方案

虽然 Redis 是内存数据库,但工具类已做足数据安全保障,避免数据丢失:

  1. 数据源兜底:未读计数的 “源头” 是 MySQL(申请记录存在 MySQL),Redis 仅为缓存;
  2. 异常修正:若 Redis 数据丢失,getUnreadCount 会返回 0,而 addUnreadCount 会重新查询 MySQL 总申请数,自动修正未读计数,不会失真。

到此这篇关于Redis实现未读消息计数的示例代码的文章就介绍到这了,更多相关Redis 未读消息计数内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

您可能感兴趣的文章:
阅读全文