《零散知识点 · Redis 实现日榜、周榜》

📢 大家好,我是 【战神刘玉栋】,有10多年的研发经验,致力于前后端技术栈的知识沉淀和传播。 💗
🌻 CSDN入驻不久,希望大家多多支持,后续会继续提升文章质量,绝不滥竽充数,欢迎多多交流。👍

CSDN.gif

Redis 实现日榜、周榜

设计方案 - 背景说明:
要实现排行榜功能,如果使用数据库来进行排行,我们大致的思路就是在这个表再定义出一个字段,如 hot 来记录热度值,然后在数据库查询时通过 order by hot 来实现排行,如果有影响排行热度的操作就直接 update 修改数据库表。
这样做的优缺点也很明显,就拿 博客热度排行 而言,可能很多操作都会影响到热度的变化。如浏览、评论、点赞等等,这也容易对数据库造成压力。
而使用 Redis 的 ZSET 类型,天然适合这一场景。

设计方案 - 键的命名:
每天的排行榜可以使用格式 daily_rank:,例如 daily_rank:2023-10-01。
周榜可以使用格式 weekly_rank:<week_number>,例如 weekly_rank:40(表示第40周)。
月榜可以使用格式 monthly_rank:,例如 monthly_rank:2023-10。

设计方案 - 功能设计:
每日排行榜更新: 每天将用户的分数更新到对应的 daily_rank: 键中。
生成周榜: 每周的最后一天(如周日),合并过去7天的排行榜生成周榜。
生成月榜: 每月的最后一天,合并过去30天的排行榜生成月榜。

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.data.redis.core.ZSetOperations;
import org.springframework.stereotype.Service;

import java.time.LocalDate;
import java.time.temporal.WeekFields;
import java.util.*;
import java.util.stream.Collectors;

@Service
public class RankService {

    @Autowired
    private RedisTemplate<String, String> redisTemplate;

    // 接受用户 ID 和分数,将其添加到当天的排行榜中。
    public void updateDailyRank(String userId, double score) {
        String today = LocalDate.now().toString();
        ZSetOperations<String, String> zSetOperations = redisTemplate.opsForZSet();
        zSetOperations.add("daily_rank:" + today, userId, score);
    }

    // 获取过去7天的排行榜,并将它们合并到当前周的周榜中。
    public void generateWeeklyRank() {
        LocalDate today = LocalDate.now();
        int weekNumber = today.get(WeekFields.of(Locale.getDefault()).weekOfWeekBasedYear());
        String weeklyRankKey = "weekly_rank:" + weekNumber;

        ZSetOperations<String, String> zSetOperations = redisTemplate.opsForZSet();
        Set<String> keys = new HashSet<>();

        // 获取过去7天的排行榜
        for (int i = 0; i < 7; i++) {
            String date = today.minusDays(i).toString();
            keys.add("daily_rank:" + date);
        }

        // 合并到周榜
        for (String key : keys) {
            zSetOperations.unionAndStore(key, weeklyRankKey);
        }
    }

    // 方法获取过去30天的排行榜,并将它们合并到当前月的月榜中。
    public void generateMonthlyRank() {
        LocalDate today = LocalDate.now();
        String monthKey = today.getYear() + "-" + String.format("%02d", today.getMonthValue());
        String monthlyRankKey = "monthly_rank:" + monthKey;

        ZSetOperations<String, String> zSetOperations = redisTemplate.opsForZSet();
        Set<String> keys = new HashSet<>();

        // 获取过去30天的排行榜
        for (int i = 0; i < 30; i++) {
            String date = today.minusDays(i).toString();
            keys.add("daily_rank:" + date);
        }

        // 合并到月榜
        for (String key : keys) {
            zSetOperations.unionAndStore(key, monthlyRankKey);
        }
    }

    //返回当前周和当前月的排行榜成员。
    public Set<String> getWeeklyRank() {
        int weekNumber = LocalDate.now().get(WeekFields.of(Locale.getDefault()).weekOfWeekBasedYear());
        return redisTemplate.opsForZSet().range("weekly_rank:" + weekNumber, 0, -1);
    }

    //返回当前周和当前月的排行榜成员。
    public Set<String> getMonthlyRank() {
        String monthKey = LocalDate.now().getYear() + "-" + String.format("%02d", LocalDate.now().getMonthValue());
        return redisTemplate.opsForZSet().range("monthly_rank:" + monthKey, 0, -1);
    }
}

总结陈词

💗 后续会逐步分享企业实际开发中的实战经验,有需要交流的可以联系博主。

CSDN_END.gif

  • 3
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论
首先,需要明确排行榜的实现方式。一般而言,排行榜可以使用有序集合(sorted set)来实现,其中成员为需要排名的对象,分数为该对象的分数(或者得分、积分等),分数越高排名越靠前。 接下来,我们针对日榜周榜和月榜分别进行实现。 ## 日榜 对于日榜,我们可以使用 Redis 的 ZINCRBY 命令来增加某个对象的分数,并使用 ZREVRANGE 命令来获取分数最高的前 N 个对象,从而得到日榜排名。 具体实现步骤如下: 1. 每当一个对象获得分数时,使用 ZINCRBY 命令将其分数增加; 2. 使用 ZREVRANGE 命令获取排序后的前 N 个对象。 例如,假设我们要维护一个日榜,其中每个用户有一个得分,我们可以使用以下代码实现: ```php // 增加用户得分 $redis->zincrby('daily_ranking', $score, $user_id); // 获取前 N 名用户 $result = $redis->zrevrange('daily_ranking', 0, $n - 1, 'WITHSCORES'); ``` 其中,$score 为用户得分,$user_id 为用户 ID,$n 为需要获取的前 N 名用户数量。 ## 周榜 对于周榜,我们可以使用 Redis 的事务(transaction)来实现。具体实现步骤如下: 1. 获取当前日期所在的周的编号; 2. 在 Redis 中创建一个键名为 "weekly_ranking:周编号" 的有序集合(sorted set); 3. 使用事务,将该周内所有用户的得分分别增加到对应的有序集合中; 4. 使用 ZREVRANGE 命令获取分数最高的前 N 个对象。 例如,假设我们要维护一个周榜,其中每个用户有一个得分,我们可以使用以下代码实现: ```php // 获取当前日期所在的周的编号 $week_number = date('W'); // 开启 Redis 事务 $redis->multi(); // 增加用户得分 $redis->zincrby("weekly_ranking:$week_number", $score, $user_id); // 获取前 N 名用户 $redis->zrevrange("weekly_ranking:$week_number", 0, $n - 1, 'WITHSCORES'); // 执行事务 $result = $redis->exec(); ``` 其中,$score 为用户得分,$user_id 为用户 ID,$n 为需要获取的前 N 名用户数量。 ## 月榜 对于月榜,与周榜类似,我们可以使用 Redis 的事务来实现。具体实现步骤如下: 1. 获取当前日期所在的月份; 2. 在 Redis 中创建一个键名为 "monthly_ranking:年份-月份" 的有序集合(sorted set); 3. 使用事务,将该月内所有用户的得分分别增加到对应的有序集合中; 4. 使用 ZREVRANGE 命令获取分数最高的前 N 个对象。 例如,假设我们要维护一个月榜,其中每个用户有一个得分,我们可以使用以下代码实现: ```php // 获取当前日期所在的月份 $month = date('Y-m'); // 开启 Redis 事务 $redis->multi(); // 增加用户得分 $redis->zincrby("monthly_ranking:$month", $score, $user_id); // 获取前 N 名用户 $redis->zrevrange("monthly_ranking:$month", 0, $n - 1, 'WITHSCORES'); // 执行事务 $result = $redis->exec(); ``` 其中,$score 为用户得分,$user_id 为用户 ID,$n 为需要获取的前 N 名用户数量。 需要注意的是,以上实现仅供参考,具体实现方式还要根据实际情况进行调整。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

战神刘玉栋

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值