首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >千万级用户ms级抽奖N名设计方案

千万级用户ms级抽奖N名设计方案

作者头像
JavaEdge
发布2022-09-27 20:15:44
3020
发布2022-09-27 20:15:44
举报
文章被收录于专栏:JavaEdgeJavaEdgeJavaEdge

1 需求

大促节零点时,从关注的用户中抽出N个人进行礼品发放,预计全网超过千万用户参加关注抽奖活动,要求:

  • 同一用户不能重复参与
  • 同一用户不允许二次中奖

2 设计方案

2.1 最原始

rand(),对每行随机产生一个随机数

select * from 关注用户表 order by rand() desc limit,0,100

预计千万级别的对技术倒排大概率凉凉。

2.2 N次随机选择SQL

效率可以,不过要先后执行两条SQL,并发时有原子性问题,且RAND函数不能保证不重复中奖。

offset = SELECT FLOOR(RAND() * COUNT(*)) AS offset from 关注用户表
select * from 关注用户表 limit offset,1

2.3 Redis Set随机弹出

step1: 在用户关注直播间在写入MySQL关注用户表时,再往Redis增加一个userlist Set,存储用户编号。 可保证用户全局唯一(避免用户反复的取消和关注影响数据记录),且数据基于Hash乱序存储,取出的直接就是随机值。

sadd userlist xxxid

预计用户编号long类型,100万50MB, 1000万用户也仅500MB。

step2: 抽奖时,直接使用spop,弹出随机的100个用户编号,该操作是原子性,先弹出再返回,在加上Redist命令队列单线程,不存在并发问题,杜绝重复中奖。

step3: 执行1次select in,提取数据,因为都是通过主键提取,效率快也不存在in索引失效问题,但要注意in的数量上限是1000个,超过1000个备选项要拆成多个in。

2.4 纯Redis

内存充足不差钱时可用。因为抽奖结果页面通常只显示用户昵称,还可使用Rdis提速,用内存换时间。

sadd userlist '123456:ikun'
sadd userlist '123456:akun'
sadd userlist '123456:bkun'

估算千万用户需3G内存,spop提取速度完全可控制在3ms内完成,且不重复。

Redis不怕Key多,只是怕大Key。测试结果:

在这里插入图片描述
在这里插入图片描述

1000次pop执行时间2565,即每次 pop 只需 2.5ms。

3 总结

你有什么更好的方案吗?

本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
原始发表:2022-09-19 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 1 需求
  • 2 设计方案
    • 2.1 最原始
      • 2.2 N次随机选择SQL
        • 2.3 Redis Set随机弹出
          • 2.4 纯Redis
          • 3 总结
          相关产品与服务
          云数据库 Redis
          腾讯云数据库 Redis(TencentDB for Redis)是腾讯云打造的兼容 Redis 协议的缓存和存储服务。丰富的数据结构能帮助您完成不同类型的业务场景开发。支持主从热备,提供自动容灾切换、数据备份、故障迁移、实例监控、在线扩容、数据回档等全套的数据库服务。
          领券
          问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档