Python Web中怎么利用Redis实现点赞与动态排行榜_基于ZSET有序集合与Hash复合数据操作

Python Web中怎么利用Redis实现点赞与动态排行榜_基于ZSET有序集合与Hash复合数据操作p p zset 适合存点赞数因其天然支持按分数排序和范围查询 而 incr hash 方案无法高效获取热度榜单 zset 的 member 应为动态 id score 为点赞数 用无条件 zadd 覆盖更新 hash 存储用户点赞状态以节省内存 直接对每个动态用 INCR 记点赞总数 再用 HASH 存用户是否点过 看似简单

大家好,我是讯享网,很高兴认识大家。这里提供最前沿的Ai技术和互联网信息。



 

zset适合存点赞数因其天然支持按分数排序和范围查询,而incr+hash方案无法高效获取热度榜单;zset的member应为动态id,score为点赞数,用无条件zadd覆盖更新,hash存储用户点赞状态以节省内存。

python web中怎么利用redis实现点赞与动态排行榜_基于zset有序集合与hash复合数据操作

直接对每个动态用 INCR 记点赞总数,再用 HASH 存用户是否点过,看似简单,但无法高效回答“当前热度前10的动态是哪些”。ZSET 天然支持按分数(如点赞数)排序和范围查询,ZREVRANGE post:zset 0 9 WITHSCORES 一行就拿到实时榜单。而纯计数方案得先查所有动态ID、再逐个 HGET 点赞数、最后在应用层排序——数据量一上去就卡住。

实操建议:

  • ZSETmember 用动态唯一 ID(如 "post:123"),score 设为当前点赞数,每次点赞/取消点赞都用 ZADD 覆盖更新
  • 不要把用户ID塞进 ZSET 成员里——那会混淆“被点赞对象”和“点赞人”,查排行榜时无法区分主体
  • 如果点赞数可能为小数(比如加权热度),ZSET 支持浮点 score,整数反而要转成 float 避免精度丢失

判断用户 A 是否已点过动态 B,最直觉是建一个 SET:如 liked:post:123SISMEMBER 查,SADD 加。但用户量大时,每个动态一个 SET 会产生海量小 key,Redis 内存碎片和元数据开销明显上升。

改用 HASH:以动态 ID 为 key,用户 ID 为 field,固定值(如 "1")为 value。例如:HSET liked:post:123 uid:456 "1"。好处是:

  • 所有用户对同一动态的点赞记录收敛到一个 key,节省 key 数量级
  • HGETSISMEMBER 少一次哈希寻址(HASH 内部是字典,field 直接定位)
  • 后续若要统计“某用户点过哪些动态”,可另建 SETZSET(如 liked:uid:456),不与主逻辑耦合

常见误区:以为点赞数增加要先 ZSCORE 查旧值,再 ZADD XX 更新。这引入竞态——两个请求并发时,可能都读到旧值,导致只加了1次而非2次。

正确做法是让业务层算出目标分数,然后无条件 ZADD

# 用户A点赞 → 当前总点赞数 = 原值 + 1 redis.zadd("post:zset", { "post:123": current_score + 1 }) 

用户B取消点赞 → 当前总点赞数 = 原值 - 1

redis.zadd("post:zset", { "post:123": current_score - 1 })

关键点:

  • ZADD 对已有 member 是原子覆盖,不存在中间态;只要业务层能保证“加1/减1”的计算逻辑一致,就无需锁
  • 别用 XX(仅更新存在 member)或 NX(仅新增),否则首次点赞可能失败——因为 ZSET 里还没这个 member
  • 如果需要严格精确(比如防刷),应在应用层做限流或校验,而不是依赖 ZADD 选项

ZREVRANGE post:zset 0 9 拿前10名很稳;但有人想“查点赞数在100~500之间的动态”,改用 ZREVRANGEBYSCORE post:zset 500 100。问题来了:如果大量动态分数相同(比如都为0),Redis 会遍历所有 score=0 的 member 直到凑够 limit,O(N) 时间复杂度,延迟飙升。

规避方式:

  • 避免用 ZREVRANGEBYSCORE 做非精确分页,尤其分数分布密集时
  • 真要按分数区间查,给 member 拼接时间戳后缀(如 "post:123:"),让相同分数的 member 也天然有序,再配合 WITHSCORES + 应用层过滤
  • 更稳妥的是:排行榜只走 ZREVRANGE + 游标(cursor),前端传上次返回的最小 score 和对应 member,后端用 ZREVRANGEBYSCORE … LIMIT 10 精确续查

复合操作本身不难,难的是各数据结构边界清晰——ZSET 只管排序依据,HASH 只管用户动作事实,别让一个结构承担两种语义。稍一混淆,扩容或排查时就会发现某个 key 内存暴涨却不知源头。

Python免费学习笔记(深入):立即使用

 
在学习笔记中,你将探索 Python 的核心概念和高级技巧!



小讯
上一篇 2026-04-14 10:52
下一篇 2026-04-14 10:50

相关推荐

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容,请联系我们,一经查实,本站将立刻删除。
如需转载请保留出处:https://51itzy.com/kjqy/261314.html