服务器之家:专注于服务器技术及软件下载分享
分类导航

Mysql|Sql Server|Oracle|Redis|MongoDB|PostgreSQL|Sqlite|DB2|mariadb|Access|数据库技术|

服务器之家 - 数据库 - Redis - Redis分布式缓存:微信抢红包解决方案

Redis分布式缓存:微信抢红包解决方案

2022-01-24 17:57小伙子vae Redis

微信抢红包已经在我们生活中很常见的场景了,特别是年底公司开年会和春节2个时间段。本文主要介绍了通过Redis实现微信抢红包功能,感兴趣的小伙伴可以了解一下

一、场景分析

微信抢红包已经在我们生活中很常见的场景了,特别是年底公司开年会和春节2个时间段,长辈领导都发红包,手都点抽筋了,也没抢到多少。

在这段时间里,对于单个群里的单个红包,qps也是上千的,对于整个微信红包系统,高峰的并发量是上亿的。

高峰的抢红包有3大特点:

  1. 包红包的人多:也就是创建红包的任务比较多,即红包系统是以单个红包的任务来区分,特点就是在高峰期红包任务多。
  2. 抢红包的人更多:当你发红包出去后,是几十甚至几百人来抢你的红包,即单红包的请求并发量大。
  3. 低延迟:当你发现红包时,要越快抢到越开心,所以要求抢红包的响应速度要快,一般1秒响应。

二、技术方案

1.包红包

先把金额拆解为小金额的红包,例如 总金额1000元,发10个,用户在点保存的时候,就自动拆解为10个随机小红包。

这里的存储就是个难题,多个金额(例如10个小金额的红包)如何存储?

Redis分布式缓存:微信抢红包解决方案

2.抢红包

高并发的抢红包时核心的关键技术,就是控制各个小红包的操作的原子性。

例如 10个红包在100人的群里被抢,10个红包被抢走一个的同时要红包的库存减1,即剩下19个。在整个过程中抢走一个和红包库存减1个是一个原子操作。

list的pop操作弹出一个元素的同时会自动从队列里面剔除该元素,它是一个原子性操作。

Redis分布式缓存:微信抢红包解决方案

三、案例实战

包红包

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
/**
 * 包红包的接口
 */
@getmapping(value = "/set")
public long setredpacket(int total, int count) {
    //拆解红包
    integer[] packet= this.splitredpacket(total,count);
    //为红包生成全局唯一id
    long n=this.incrementid();
    //采用list存储红包
    string key=red_packet_key+n;
    this.redistemplate.opsforlist().leftpushall(key,packet);
    //设置3天过期
    this.redistemplate.expire(key,3, timeunit.days);
    log.info("拆解红包{}={}",key,packet);
    return n;
}

拆解红包

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
/**
 * 拆解红包
 * 1.红包金额要被全部拆解完
 * 2.红包金额不能差太离谱
 * total 红包金额
 * count 红包数量
 */
public  integer[] splitredpacket(int total, int count) {
    int use = 0;
    integer[] array = new integer[count];
    random random = new random();
    for (int i = 0; i < count; i++) {
        if (i == count - 1)
            array[i] = total - use;
        else {
            // 红包随机金额浮动系数
            int avg = (total - use) * 2 / (count - i);
            array[i] = 1 + random.nextint(avg - 1);
        }
        use = use + array[i];
    }
    return array;
}

抢红包 

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
/**
 * 抢红包接口
 */
@getmapping(value = "/rob")
public int rob(long redid,long userid) {
    //第一步:验证该用户是否抢过
    object packet=this.redistemplate.opsforhash().get(red_packet_consume_key+redid,string.valueof(userid));
    if(packet==null){
        //第二步:从list队列,弹出一个红包
        object obj=this.redistemplate.opsforlist().leftpop(red_packet_key+redid);
        if(obj!=null){
            //第三步:抢到红包存起来
            this.redistemplate.opsforhash().put(red_packet_consume_key+redid,string.valueof(userid),obj);
            log.info("用户={}抢到{}",userid,obj);
            //todo 异步把数据落地到数据库上
            
            return (integer) obj;
        }
        //-1 代表抢完
        return -1;
    }
    //-2 代表已抢
    return -2;
}

到此这篇关于redis分布式缓存:微信抢红包解决方案的文章就介绍到这了,更多相关redis微信抢红包内容请搜索服务器之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持服务器之家!

原文链接:https://juejin.cn/post/7039876451058843656

延伸 · 阅读

精彩推荐
  • Redisredis实现排行榜功能

    redis实现排行榜功能

    排行榜在很多地方都能使用到,redis的zset可以很方便地用来实现排行榜功能,本文就来简单的介绍一下如何使用,具有一定的参考价值,感兴趣的小伙伴们...

    乘月归5022021-08-05
  • Redis详解Redis复制原理

    详解Redis复制原理

    与大多数db一样,Redis也提供了复制机制,以满足故障恢复和负载均衡等需求。复制也是Redis高可用的基础,哨兵和集群都是建立在复制基础上实现高可用的...

    李留广10222021-08-09
  • RedisRedis全量复制与部分复制示例详解

    Redis全量复制与部分复制示例详解

    这篇文章主要给大家介绍了关于Redis全量复制与部分复制的相关资料,文中通过示例代码介绍的非常详细,对大家学习或者使用Redis爬虫具有一定的参考学习...

    豆子先生5052019-11-27
  • RedisRedis如何实现数据库读写分离详解

    Redis如何实现数据库读写分离详解

    Redis的主从架构,能帮助我们实现读多,写少的情况,下面这篇文章主要给大家介绍了关于Redis如何实现数据库读写分离的相关资料,文中通过示例代码介绍...

    罗兵漂流记6092019-11-11
  • RedisRedis的配置、启动、操作和关闭方法

    Redis的配置、启动、操作和关闭方法

    今天小编就为大家分享一篇Redis的配置、启动、操作和关闭方法,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧 ...

    大道化简5312019-11-14
  • RedisRedis 事务知识点相关总结

    Redis 事务知识点相关总结

    这篇文章主要介绍了Redis 事务相关总结,帮助大家更好的理解和学习使用Redis,感兴趣的朋友可以了解下...

    AsiaYe8232021-07-28
  • Redisredis中如何使用lua脚本让你的灵活性提高5个逼格详解

    redis中如何使用lua脚本让你的灵活性提高5个逼格详解

    这篇文章主要给大家介绍了关于redis中如何使用lua脚本让你的灵活性提高5个逼格的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具...

    一线码农5812019-11-18
  • Redisredis 交集、并集、差集的具体使用

    redis 交集、并集、差集的具体使用

    这篇文章主要介绍了redis 交集、并集、差集的具体使用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友...

    xiaojin21cen10152021-07-27