新技能get√丨抢了那么多红包,你知道算法是怎么实现的吗

2023-10-24 08:50

本文主要是介绍新技能get√丨抢了那么多红包,你知道算法是怎么实现的吗,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

没红包可抢的日子真是太难熬了


好在元宵节到了!


微信官方统计,仅除夕到初五,微信总计收发红包次数多达321亿次,上百亿个微信红包散落在了万家灯火的欢声笑语之中,这其中,就“一次能抢到多少钱”的这个问题,发生了这样一段“从线上到线下”的对话:

妈妈:老盯着手机干什么?

儿子:抢红包

妈妈:能抢多少?

儿子:看手法,有时候白忙,有时候几分钱到一两块

妈妈:我给你10块,你去给我把锅刷了…



看到这个段子,玩营销的在琢磨商机,做设计的在审视红包配色,做产品的在考虑互动机制,而我们超爱思考的程序员们,早已就红包算法的n种实现方法讨论得热火朝天……


本周起,小卓每周一与各位IT大侠讨论各种有趣的技术问题,今天,咱们先聊聊如火如荼的“红包算法”。


MONDAY:新技能get

微信红包算法



首先,为了采集样本,顺带感谢毕业多年来一直孜孜不倦嘘寒问暖的大学同学们,小卓在同学群里发了个红包:设定红包总额为10元,支持28个人随机领取(礼轻情意重啊)。

不出所料,大家一边骂我抠门,一边抢抢抢。


在挨骂的同时,小卓没有忘记发红包的初衷,开始兢兢业业地采集样本:

A领取了0.26元

B领取了0.29元

C领取了0.02元

D领取了0.56元

E领取了0.64元

……

有趣的现象出现了:随机分配,人人有份,又各自不同,微信是采用什么样的算法做到的?

简单百度了下,目前尚未有官方的说明,仅仅在知乎里有一个较为热门的讨论《微信红包的随机算法是怎样实现的?》(https://www.zhihu.com/question/22625187 )不过他们讨论的太过于深入,有掉坑之嫌,于是自己动手。


按照红包的逻辑,红包算法需要满足以下几点要求:

    1、每个人都要能够领取到红包;

    2、每个人领取到的红包金额总和=总金额;

    3、每个人领取到的红包金额不等,但也不能差的太离谱,不然就没趣味;

    4、算法一定要简单,不然对不起腾讯这个招牌;


正式编码之前,先搭建一个递进的模型来分析规律:

设定总金额为10元,有N个人随机领取:

N=1

则红包金额=X元; 

N=2,

为保证第二个红包可以正常发出,

红包1=0.01至9.99之间的某个随机数,

红包2=10-第一个红包金额; 

N=3,

红包1=0.01至9.98之间的某个随机数,

红包2=0.01至(10-红包1-0.01)的某个随机数,

红包3=10-红包1-红包2 

……


至此,规律出现啦!开始编码!

header("Content-Type: text/html;charset=utf-8");//输出不乱码,你懂的

$total=10;//红包总额

$num=8;// 分成8个红包,支持8人随机领取

$min=0.01;//每个人最少能收到0.01元

for ($i=1;$i<$num;$i++)

{

    $safe_total=$total-($num-$i)*$min;//随机安全上限

    $money=mt_rand($min*100,$safe_total*100)/100;

    $total=$total-$money;

    echo '第'.$i.'个红包:'.$money.' 元,余额:'.$total.' 元 <br/>';

}

echo '第'.$num.'个红包:'.$total.' 元,余额:0 元';


输入一看,波动太大,这数据太没意思了!

第1个红包:7.48 元,余额:2.52 元 

第2个红包:1.9 元,余额:0.62 元 

第3个红包:0.49 元,余额:0.13 元 

第4个红包:0.04 元,余额:0.09 元 

第5个红包:0.03 元,余额:0.06 元 

第6个红包:0.03 元,余额:0.03 元 

第7个红包:0.01 元,余额:0.02 元 

第8个红包:0.02 元,余额:0 元


改良一下,将平均值作为随机安全上限来控制波动差

header("Content-Type: text/html;charset=utf-8");

$total=10;//红包总额

$num=8;// 分成8个红包,支持8人随机领取

$min=0.01;//每个人最少能收到0.01元

for ($i=1;$i<$num;$i++)

{

    $safe_total=($total-($num-$i)*$min)/($num-$i);//随机安全上限

    $money=mt_rand($min*100,$safe_total*100)/100;

    $total=$total-$money;

    echo '第'.$i.'个红包:'.$money.' 元,余额:'.$total.' 元 <br/>';

}

echo '第'.$num.'个红包:'.$total.' 元,余额:0 元';


输出结果: 

第1个红包:0.06 元,余额:9.94 元 

第2个红包:1.55 元,余额:8.39 元 

第3个红包:0.25 元,余额:8.14 元 

第4个红包:0.98 元,余额:7.16 元 

第5个红包:1.88 元,余额:5.28 元 

第6个红包:1.92 元,余额:3.36 元 

第7个红包:2.98 元,余额:0.38 元 

第8个红包:0.38 元,余额:0 元


就这样,实现了一种红包算法。


什么?你想到了更好的?赶快留言啊,说来听听!

算法来自:http://blog.cqcoder.com/微信红包的算法实现探讨/



这篇关于新技能get√丨抢了那么多红包,你知道算法是怎么实现的吗的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/274067

相关文章

SpringBoot集成redisson实现延时队列教程

《SpringBoot集成redisson实现延时队列教程》文章介绍了使用Redisson实现延迟队列的完整步骤,包括依赖导入、Redis配置、工具类封装、业务枚举定义、执行器实现、Bean创建、消费... 目录1、先给项目导入Redisson依赖2、配置redis3、创建 RedissonConfig 配

Python的Darts库实现时间序列预测

《Python的Darts库实现时间序列预测》Darts一个集统计、机器学习与深度学习模型于一体的Python时间序列预测库,本文主要介绍了Python的Darts库实现时间序列预测,感兴趣的可以了解... 目录目录一、什么是 Darts?二、安装与基本配置安装 Darts导入基础模块三、时间序列数据结构与

Python使用FastAPI实现大文件分片上传与断点续传功能

《Python使用FastAPI实现大文件分片上传与断点续传功能》大文件直传常遇到超时、网络抖动失败、失败后只能重传的问题,分片上传+断点续传可以把大文件拆成若干小块逐个上传,并在中断后从已完成分片继... 目录一、接口设计二、服务端实现(FastAPI)2.1 运行环境2.2 目录结构建议2.3 serv

C#实现千万数据秒级导入的代码

《C#实现千万数据秒级导入的代码》在实际开发中excel导入很常见,现代社会中很容易遇到大数据处理业务,所以本文我就给大家分享一下千万数据秒级导入怎么实现,文中有详细的代码示例供大家参考,需要的朋友可... 目录前言一、数据存储二、处理逻辑优化前代码处理逻辑优化后的代码总结前言在实际开发中excel导入很

SpringBoot+RustFS 实现文件切片极速上传的实例代码

《SpringBoot+RustFS实现文件切片极速上传的实例代码》本文介绍利用SpringBoot和RustFS构建高性能文件切片上传系统,实现大文件秒传、断点续传和分片上传等功能,具有一定的参考... 目录一、为什么选择 RustFS + SpringBoot?二、环境准备与部署2.1 安装 RustF

Nginx部署HTTP/3的实现步骤

《Nginx部署HTTP/3的实现步骤》本文介绍了在Nginx中部署HTTP/3的详细步骤,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学... 目录前提条件第一步:安装必要的依赖库第二步:获取并构建 BoringSSL第三步:获取 Nginx

MyBatis Plus实现时间字段自动填充的完整方案

《MyBatisPlus实现时间字段自动填充的完整方案》在日常开发中,我们经常需要记录数据的创建时间和更新时间,传统的做法是在每次插入或更新操作时手动设置这些时间字段,这种方式不仅繁琐,还容易遗漏,... 目录前言解决目标技术栈实现步骤1. 实体类注解配置2. 创建元数据处理器3. 服务层代码优化填充机制详

Python实现Excel批量样式修改器(附完整代码)

《Python实现Excel批量样式修改器(附完整代码)》这篇文章主要为大家详细介绍了如何使用Python实现一个Excel批量样式修改器,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一... 目录前言功能特性核心功能界面特性系统要求安装说明使用指南基本操作流程高级功能技术实现核心技术栈关键函

Java实现字节字符转bcd编码

《Java实现字节字符转bcd编码》BCD是一种将十进制数字编码为二进制的表示方式,常用于数字显示和存储,本文将介绍如何在Java中实现字节字符转BCD码的过程,需要的小伙伴可以了解下... 目录前言BCD码是什么Java实现字节转bcd编码方法补充总结前言BCD码(Binary-Coded Decima

SpringBoot全局域名替换的实现

《SpringBoot全局域名替换的实现》本文主要介绍了SpringBoot全局域名替换的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一... 目录 项目结构⚙️ 配置文件application.yml️ 配置类AppProperties.Ja