新技能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

相关文章

MyBatis-Plus逻辑删除实现过程

《MyBatis-Plus逻辑删除实现过程》本文介绍了MyBatis-Plus如何实现逻辑删除功能,包括自动填充字段、配置与实现步骤、常见应用场景,并展示了如何使用remove方法进行逻辑删除,逻辑删... 目录1. 逻辑删除的必要性编程1.1 逻辑删除的定义1.2 逻辑删php除的优点1.3 适用场景2.

C#借助Spire.XLS for .NET实现在Excel中添加文档属性

《C#借助Spire.XLSfor.NET实现在Excel中添加文档属性》在日常的数据处理和项目管理中,Excel文档扮演着举足轻重的角色,本文将深入探讨如何在C#中借助强大的第三方库Spire.... 目录为什么需要程序化添加Excel文档属性使用Spire.XLS for .NET库实现文档属性管理Sp

Python+FFmpeg实现视频自动化处理的完整指南

《Python+FFmpeg实现视频自动化处理的完整指南》本文总结了一套在Python中使用subprocess.run调用FFmpeg进行视频自动化处理的解决方案,涵盖了跨平台硬件加速、中间素材处理... 目录一、 跨平台硬件加速:统一接口设计1. 核心映射逻辑2. python 实现代码二、 中间素材处

Java数组动态扩容的实现示例

《Java数组动态扩容的实现示例》本文主要介绍了Java数组动态扩容的实现示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录1 问题2 方法3 结语1 问题实现动态的给数组添加元素效果,实现对数组扩容,原始数组使用静态分配

Python实现快速扫描目标主机的开放端口和服务

《Python实现快速扫描目标主机的开放端口和服务》这篇文章主要为大家详细介绍了如何使用Python编写一个功能强大的端口扫描器脚本,实现快速扫描目标主机的开放端口和服务,感兴趣的小伙伴可以了解下... 目录功能介绍场景应用1. 网络安全审计2. 系统管理维护3. 网络故障排查4. 合规性检查报错处理1.

Python轻松实现Word到Markdown的转换

《Python轻松实现Word到Markdown的转换》在文档管理、内容发布等场景中,将Word转换为Markdown格式是常见需求,本文将介绍如何使用FreeSpire.DocforPython实现... 目录一、工具简介二、核心转换实现1. 基础单文件转换2. 批量转换Word文件三、工具特性分析优点局

Springboot3统一返回类设计全过程(从问题到实现)

《Springboot3统一返回类设计全过程(从问题到实现)》文章介绍了如何在SpringBoot3中设计一个统一返回类,以实现前后端接口返回格式的一致性,该类包含状态码、描述信息、业务数据和时间戳,... 目录Spring Boot 3 统一返回类设计:从问题到实现一、核心需求:统一返回类要解决什么问题?

Java使用Spire.Doc for Java实现Word自动化插入图片

《Java使用Spire.DocforJava实现Word自动化插入图片》在日常工作中,Word文档是不可或缺的工具,而图片作为信息传达的重要载体,其在文档中的插入与布局显得尤为关键,下面我们就来... 目录1. Spire.Doc for Java库介绍与安装2. 使用特定的环绕方式插入图片3. 在指定位

Java使用Spire.Barcode for Java实现条形码生成与识别

《Java使用Spire.BarcodeforJava实现条形码生成与识别》在现代商业和技术领域,条形码无处不在,本教程将引导您深入了解如何在您的Java项目中利用Spire.Barcodefor... 目录1. Spire.Barcode for Java 简介与环境配置2. 使用 Spire.Barco

Java利用Spire.Doc for Java实现在模板的基础上创建Word文档

《Java利用Spire.DocforJava实现在模板的基础上创建Word文档》在日常开发中,我们经常需要根据特定数据动态生成Word文档,本文将深入探讨如何利用强大的Java库Spire.Do... 目录1. Spire.Doc for Java 库介绍与安装特点与优势Maven 依赖配置2. 通过替换