设为首页收藏本站
网站公告 | 这是第一条公告
     

 找回密码
 立即注册
缓存时间21 现在时间21 缓存数据 你是我生命中所能经历的,最最深切的感觉。

你是我生命中所能经历的,最最深切的感觉。

查看: 495|回复: 0

使用Redis实现实时排行榜的示例

[复制链接]

  离线 

TA的专栏

  • 打卡等级:即来则安
  • 打卡总天数:20
  • 打卡月天数:0
  • 打卡总奖励:231
  • 最近打卡:2025-10-23 08:57:19
等级头衔

等級:晓枫资讯-上等兵

在线时间
0 小时

积分成就
威望
0
贡献
320
主题
288
精华
0
金钱
1161
积分
648
注册时间
2023-2-11
最后登录
2025-10-23

发表于 2025-9-1 07:22:14 | 显示全部楼层 |阅读模式
为了实现一个实时排行榜系统,我们可以使用Redis的有序集合(ZSet),其底层通常是使用跳跃表实现的。有序集合允许我们按照分数(score)对成员(member)进行排序,因此非常适合用来实现排行榜。本文首先介绍有序集合及其底层数据结构——跳表,然后使用Python和Redis结合,展示一个简单的排行榜系统。

一、ZSet 概述


1.1 ZSet 介绍

实现一个排行榜,很多人可能首先想到的是使用MySQL的
  1. order by
复制代码
来排序。然而,当数据量达到百万级别时,使用数据库排序的代价是很大的。因此,Redis的有序集合(ZSet)成为了一个更好的选择。
ZSet(Sorted Set)的特点如下:

  • 唯一性:集合内的元素(成员)是唯一的。
  • 有序性:与普通Set的无序性不同,ZSet的成员是“有序的”,这种有序性是基于成员所关联的“分数”(score)进行排序的,分数是浮点类型。

1.2 Zset 底层原理

ZSet 是Redis中的一种复杂数据结构,它在Set的基础上增加了一个权重参数score,使得集合中的元素能按score进行有序排列。
ZSet的底层实现通常有两种数据结构:

  • 当元素数量较少或元素长度较短时,采用压缩列表(ziplist)
  • 当元素数量达到一定量或者元素长度超过一定限制时,采用跳跃表(skiplist)
跳表(skiplist)具有多层链表结构,查询、插入和删除操作的平均时间复杂度均为O(log n)。

1.3 ZSet 主要操作命令


    1. ZADD key score member
    复制代码
    :将元素及其分数添加到有序集合中。
    1. ZINCRBY key increment member
    复制代码
    :为有序集合中的元素增加或减少分数。
    1. ZRANGE key start stop [WITHSCORES]
    复制代码
    :获取有序集合中分数从小到大的排名在指定范围内的成员。
    1. ZREVRANGE key start stop [WITHSCORES]
    复制代码
    :获取有序集合中分数从大到小的排名在指定范围内的成员。
    1. ZRANK key member
    复制代码
    :获取成员在有序集合中的排名(从小到大的排名,排名从0开始)。
    1. ZREVRANK key member
    复制代码
    :获取成员在有序集合中的排名(从大到小的排名,排名从0开始)。
    1. ZSCORE key member
    复制代码
    :获取成员在有序集合中的分数。
    1. ZCARD key
    复制代码
    :获取有序集合的基数,即成员数量。

二、使用 Redis 和 Python 实现实时排行榜

下面是一个使用Python的
  1. redis
复制代码
库来操作ZSet并实现实时排行榜的示例。

2.1 安装所需的库

首先确保已经安装
  1. redis
复制代码
库:
  1. pip install redis
复制代码
2.2 初始化RedisLeaderboard类

接下来,我们实现一个
  1. RedisLeaderboard
复制代码
类来管理排行榜:
  1. import redis
  2. from flask import Flask, render_template
  3. import sys

  4. app = Flask(__name__)

  5. # Initialize Redis connection with error handling
  6. try:
  7.     r = redis.Redis(
  8.         host='192.168.88.139',
  9.         password='123456',
  10.         port=6379,
  11.         db=0,
  12.         socket_connect_timeout=3,  # 3 seconds timeout
  13.         decode_responses=True  # Automatically decode responses to UTF-8
  14.     )
  15.     # Test the connection
  16.     r.ping()
  17.     print("成功连接Redis", file=sys.stderr)
  18. except redis.ConnectionError as e:
  19.     print(f"连接Redis失败: {e}", file=sys.stderr)
  20.     r = None  # Set to None so we can check later


  21. @app.route('/')
  22. def leaderboard():
  23.     if r is None:
  24.         return render_template('error.html',
  25.                                message="Redis server is not available"), 503

  26.     try:
  27.         top_10 = get_top_n(10)
  28.         return render_template('leaderboard.html', leaderboard=top_10)
  29.     except redis.RedisError as e:
  30.         return render_template('error.html',
  31.                                message=f"Redis error: {str(e)}"), 500


  32. def get_top_n(n):
  33.     try:
  34.         top_n = r.zrevrange("game_leaderboard", 0, n - 1, withscores=True)
  35.         leaderboard = []
  36.         for rank, (user_id, score) in enumerate(top_n, start=1):
  37.             leaderboard.append({
  38.                 "rank": rank,
  39.                 "user_id": user_id,  # No need to decode with decode_responses=True
  40.                 "score": float(score)
  41.             })
  42.         return leaderboard
  43.     except redis.RedisError as e:
  44.         print(f"Redis operation failed: {e}", file=sys.stderr)
  45.         raise  # Re-raise the exception to be handled by the route


  46. if __name__ == '__main__':
  47.     app.run(debug=True)
复制代码
1.png


2.3 案例数据
  1. import redis

  2. r = redis.Redis(host='192.168.88.139', password='123456', port=6379, db=0)


  3. def add_score(user_id, score):
  4.     r.zadd("game_leaderboard", {user_id: score})


  5. def update_score(user_id, score):
  6.     r.zincrby("game_leaderboard", score, user_id)


  7. def get_top_n(n):
  8.     top_n = r.zrevrange("game_leaderboard", 0, n - 1, withscores=True)
  9.     leaderboard = []
  10.     for rank, (user_id, score) in enumerate(top_n, start=1):
  11.         leaderboard.append({
  12.             "rank": rank,
  13.             "user_id": user_id.decode("utf-8"),
  14.             "score": score
  15.         })
  16.     return leaderboard


  17. def get_user_rank_and_score(user_id):
  18.     rank = r.zrevrank("game_leaderboard", user_id)
  19.     if rank is not None:
  20.         rank += 1
  21.     score = r.zscore("game_leaderboard", user_id)
  22.     return rank, score


  23. if __name__ == '__main__':
  24.     # 添加初始得分
  25.     add_score('user1', 100)
  26.     add_score('user2', 150)
  27.     add_score('user3', 50)

  28.     # 更新得分(加分操作),如果用户不存在,会将其得分初始化为该值
  29.     update_score('user1', 30)
  30.     update_score('user2', 20)
  31.     update_score('user3', -10)

  32.     # 获取前2名的用户
  33.     top_2 = get_top_n(2)
  34.     for entry in top_2:
  35.         print(f"Rank {entry['rank']}: UserID: {entry['user_id']} with score {entry['score']}")

  36.     # 获取特定用户的排名和得分
  37.     rank, score = get_user_rank_and_score('user1')
  38.     if rank is not None and score is not None:
  39.         print(f"User user1 is ranked {rank} with a score of {score}.")
  40.     else:
  41.         print("User user1 is not found in the leaderboard.")
复制代码
2.4 前端

需要创建一个templates文件夹,并在其中存放leaderboard.html文件:
  1. <!DOCTYPE html>
  2. <html lang="en">
  3. <head>
  4.     <meta charset="UTF-8">
  5.     <meta name="viewport" content="width=device-width, initial-scale=1.0">
  6.     <title>Leaderboard</title>
  7.     <style>
  8.         table {
  9.             width: 100%;
  10.             border-collapse: collapse;
  11.         }
  12.         th, td {
  13.             border: 1px solid black;
  14.             padding: 8px;
  15.             text-align: left;
  16.         }
  17.     </style>
  18. </head>
  19. <body>
  20.     <h1>Leaderboard</h1>
  21.     <table>
  22.         <thead>
  23.             <tr>
  24.                 <th>Rank</th>
  25.                 <th>User ID</th>
  26.                 <th>Score</th>
  27.             </tr>
  28.         </thead>
  29.         <tbody>
  30.             {% for entry in leaderboard %}
  31.             <tr>
  32.                 <td>{{ entry.rank }}</td>
  33.                 <td>{{ entry.user_id }}</td>
  34.                 <td>{{ entry.score }}</td>
  35.             </tr>
  36.             {% endfor %}
  37.         </tbody>
  38.     </table>
  39. </body>
  40. </html>
复制代码
三、结论

Redis的有序集合(ZSet)由于其高效的插入、删除、查询及排序操作,是实现实时排行榜的理想选择。跳表作为ZSet的底层数据结构之一,保证了这些操作的时间复杂度为O(log n)。结合Python的
  1. redis
复制代码
库,可以快速实现一个功能强大、高效的实时排行榜系统。
这种排行榜实现方案非常适合用于在线游戏、社交平台等各种应用场景。
到此这篇关于使用Redis实现实时排行榜的示例的文章就介绍到这了,更多相关Redis 实时排行榜内容请搜索晓枫资讯以前的文章或继续浏览下面的相关文章希望大家以后多多支持晓枫资讯!

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!
晓枫资讯-科技资讯社区-免责声明
免责声明:以上内容为本网站转自其它媒体,相关信息仅为传递更多信息之目的,不代表本网观点,亦不代表本网站赞同其观点或证实其内容的真实性。
      1、注册用户在本社区发表、转载的任何作品仅代表其个人观点,不代表本社区认同其观点。
      2、管理员及版主有权在不事先通知或不经作者准许的情况下删除其在本社区所发表的文章。
      3、本社区的文章部分内容可能来源于网络,仅供大家学习与参考,如有侵权,举报反馈:点击这里给我发消息进行删除处理。
      4、本社区一切资源不代表本站立场,并不代表本站赞同其观点和对其真实性负责。
      5、以上声明内容的最终解释权归《晓枫资讯-科技资讯社区》所有。
http://bbs.yzwlo.com 晓枫资讯--游戏IT新闻资讯~~~
严禁发布广告,淫秽、色情、赌博、暴力、凶杀、恐怖、间谍及其他违反国家法律法规的内容。!晓枫资讯-社区
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|晓枫资讯--科技资讯社区 本站已运行

CopyRight © 2022-2025 晓枫资讯--科技资讯社区 ( BBS.yzwlo.com ) . All Rights Reserved .

晓枫资讯--科技资讯社区

本站内容由用户自主分享和转载自互联网,转载目的在于传递更多信息,并不代表本网赞同其观点和对其真实性负责。

如有侵权、违反国家法律政策行为,请联系我们,我们会第一时间及时清除和处理! 举报反馈邮箱:点击这里给我发消息

Powered by Discuz! X3.5

快速回复 返回顶部 返回列表