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

相关文章

Go语言开发实现查询IP信息的MCP服务器

《Go语言开发实现查询IP信息的MCP服务器》随着MCP的快速普及和广泛应用,MCP服务器也层出不穷,本文将详细介绍如何在Go语言中使用go-mcp库来开发一个查询IP信息的MCP... 目录前言mcp-ip-geo 服务器目录结构说明查询 IP 信息功能实现工具实现工具管理查询单个 IP 信息工具的实现服

SpringBoot基于配置实现短信服务策略的动态切换

《SpringBoot基于配置实现短信服务策略的动态切换》这篇文章主要为大家详细介绍了SpringBoot在接入多个短信服务商(如阿里云、腾讯云、华为云)后,如何根据配置或环境切换使用不同的服务商,需... 目录目标功能示例配置(application.yml)配置类绑定短信发送策略接口示例:阿里云 & 腾

python实现svg图片转换为png和gif

《python实现svg图片转换为png和gif》这篇文章主要为大家详细介绍了python如何实现将svg图片格式转换为png和gif,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录python实现svg图片转换为png和gifpython实现图片格式之间的相互转换延展:基于Py

Python利用ElementTree实现快速解析XML文件

《Python利用ElementTree实现快速解析XML文件》ElementTree是Python标准库的一部分,而且是Python标准库中用于解析和操作XML数据的模块,下面小编就来和大家详细讲讲... 目录一、XML文件解析到底有多重要二、ElementTree快速入门1. 加载XML的两种方式2.

Java的栈与队列实现代码解析

《Java的栈与队列实现代码解析》栈是常见的线性数据结构,栈的特点是以先进后出的形式,后进先出,先进后出,分为栈底和栈顶,栈应用于内存的分配,表达式求值,存储临时的数据和方法的调用等,本文给大家介绍J... 目录栈的概念(Stack)栈的实现代码队列(Queue)模拟实现队列(双链表实现)循环队列(循环数组

C++如何通过Qt反射机制实现数据类序列化

《C++如何通过Qt反射机制实现数据类序列化》在C++工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作,所以本文就来聊聊C++如何通过Qt反射机制实现数据类序列化吧... 目录设计预期设计思路代码实现使用方法在 C++ 工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作。由于数据类

Python实现图片分割的多种方法总结

《Python实现图片分割的多种方法总结》图片分割是图像处理中的一个重要任务,它的目标是将图像划分为多个区域或者对象,本文为大家整理了一些常用的分割方法,大家可以根据需求自行选择... 目录1. 基于传统图像处理的分割方法(1) 使用固定阈值分割图片(2) 自适应阈值分割(3) 使用图像边缘检测分割(4)

Android实现在线预览office文档的示例详解

《Android实现在线预览office文档的示例详解》在移动端展示在线Office文档(如Word、Excel、PPT)是一项常见需求,这篇文章为大家重点介绍了两种方案的实现方法,希望对大家有一定的... 目录一、项目概述二、相关技术知识三、实现思路3.1 方案一:WebView + Office Onl

C# foreach 循环中获取索引的实现方式

《C#foreach循环中获取索引的实现方式》:本文主要介绍C#foreach循环中获取索引的实现方式,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录一、手动维护索引变量二、LINQ Select + 元组解构三、扩展方法封装索引四、使用 for 循环替代

Spring Security+JWT如何实现前后端分离权限控制

《SpringSecurity+JWT如何实现前后端分离权限控制》本篇将手把手教你用SpringSecurity+JWT搭建一套完整的登录认证与权限控制体系,具有很好的参考价值,希望对大家... 目录Spring Security+JWT实现前后端分离权限控制实战一、为什么要用 JWT?二、JWT 基本结构