文章目录

  • 问题抽象
  • 问题简化
  • 思路一
  • 思路二
  • 思路三
  • 结束语

“叮咚”,微信提示音响起,打开手机发现“相亲相爱一家人”的群里收到一个红包,天不负我,这一次终于抢到的金额终于说得过去了,虽然不是“运气王”,但有时候做个榜二也是一件很幸福的事情。

  • 由红包引发的思考
    红包的金额是如何划分的?红包的金额是怎么实现的完全随机呢?

问题抽象

将M元(M可以是两位数小数)的红包完全随机划分,最小分割单位为0.01元,分割成N份,且需要保证每人最少可领取到0.01元。

问题简化

将M乘100,最小分割单位变为1,N不变,这个问题就变成了一个纯整数的运算。

将M元(整数)的红包完全随机划分,最小分割单位为1元,分割成N份,且需要保证每人最少可领取到1元。

数据约定:(0 < M <= 10000; 0 < N <= M)

思路一

假设当前还有n个人未领取,红包剩余金额为m元
剩余的n个人按顺序领取,每个人可领取[1 ~(m - (n - 1))]区间的随机金额。
(m - (n - 1))用来确保剩余的n-1人最少可以领取到1元红包。

代码实现:

private static void randomRedPacket(int m, int n, int[] result){//当前已经有多少人领取过红包int index = result.length - n;//如果是最后一个人,则将红包剩余金额全部给这个人,并结束循环if(index == result.length - 1){result[index] = m;return;}//从剩下的金额中分配随机金额int randomValue = (int) (Math.random() * (m - (n - 1)) + 1);result[index] = randomValue;randomRedPacket(m - randomValue, n - 1, result);
}

运行结果:

解析:

当两个人平分红包的时候,看不出什么问题,当参与瓜分红包的人数多了之后,发现实际结果与期望的有所偏差.
以10人瓜分百元红包为例,回顾一下我们写的算法,其中的问题还是比较容易被发现的。
第一个抢红包的人,他的随机区间为[1, 91]元(给剩下的9人每人预留一元),在第一人抢到41元之后,第二个人的随机区间变为了[1, 51]元,以此类推,在第二个人抢到49元,第三个人抢到3元之后,从第四个人开始,他们的随机区间变为了[1, 1],排在后边的人没有了选择,只能领取保底的一元了。
由此可见,这个算法并没有实现完全的随机,排在前边的人,他们的可选区间越大,抢到大红包的可能性越大,而排在后边的人,他们的可选区间受前人影响,在前人的大快朵颐之后,只能吃些残渣剩饭了。

思路二

认识到思路一的问题,我们不妨做一些改进。
N个人,每人先领取1元。剩余的(M - N)元分成(M - N)份,依次随机分发给N个人当中的一个。

代码实现:

private static void randomRedPacket(int m, int n, int[] result){//每人先领取一元低保Arrays.fill(result, 1);//将剩下的m-n元,随机分配个其中的一个人for (int i = 0; i < m - n; i++) {result[(int) (Math.random() * n)] += 1;}
}

运行结果:

解析:

OK!看似没什么问题了,实现了对第一个领取的人和最后领取的人的公平性,红包的金额已经完全不受领取顺序影响了。
但是,细心一点的话还会发现,这个算法并不完美,最终生成的红包金额都比较趋近于平均值,2人瓜分百元红包,生成的两个红包都比较趋近于50元,5人瓜分百元红包,每个红包的金额都比较趋近于20元,10人瓜分,每个红包都比较趋近于10元。如此一来,就失去了拼手气红包的乐趣。
那么,问题出在哪里呢?
结合数学当中的概率,将N个球放到n个箱子里,求其中的一个箱子里有m个球的概率。
由此发现,每个人分得平均值的概率最大,越偏离平均值的概率越小。

思路三

N个人,每人先领取1元。
剩余的(M - N)元分成(M - N)份,在[0, (M - N)]中生成(N - 1)个随机数作为随机位置。用这(N - 1)个随机位置将剩余的(M - N)元分成N份,依次分发给每个人。

代码实现:

private static void randomRedPacket(int m, int n, int[] result){//每人先领取一元低保Arrays.fill(result, 1);//初始化一个位置数组,用于存储随机位置int[] indexArr = new int[n - 1];int lastMoney = m - n;for (int i = 0; i < n - 1; i++) {int index = (int) (Math.random() * lastMoney);indexArr[i] = index;}//对分段的位置按照从小打到排序Arrays.sort(indexArr);//将n-1个位置切割成的n段,依次分配给n个人for (int i = 0; i < result.length; i++) {if(i == 0){result[i] += indexArr[i];}else if(i == n - 1){result[i] += lastMoney - indexArr[i - 1];}else{result[i] += indexArr[i] - indexArr[i - 1];}}
}

运行结果:

解析:

终于!出现了我们想要的结果,2个人抢个百元红包你即可能抢到1元,也可能抢到99元,还可能抢到其他任意金额,开红包的过程充满了期待,充满了惊喜。

调用入口:

public static void main(String[] args) {final int[] N = new int[]{2, 5, 10};final int M = 100;for (int n : N) {final int[] result = new int[n];System.out.printf("%s个人瓜分%s元红包!\n", n, M);randomRedPacket3(M, n, result);for (int i : result) {System.out.print(i + "\t");}System.out.println();}
}

结束语

忘忧的个人公众号,欢迎大家一起交流:算法之灵魂拷问

生活就像一盒巧克力,你永远不知道下一颗会是什么味道!

解析拼手气红包金额划分算法相关推荐

  1. php 固定人数拼手气_PHP 拼手气红包 分配红包金额

    /** * 拼手气红包金额生成 * @param $money_total红包总金额 * @param $num红包个数 * @return $rand_money_arr每个红包的金额 */ fun ...

  2. java 红包算法_JAVA实现拼手气红包算法

    实现拼手气红包算法,有以下几个需要注意的地方: 抢红包的期望收益应与先后顺序无关 保证每个用户至少能抢到一个预设的最小金额,人民币红包设置的最小金额一般是0.01元,如果需要发其他货币类型的红包,比如 ...

  3. 二倍均值随机算法之抢拼手气红包场景应用

    拼手气类的游戏,更能激发用户购物和社交的趣味性,以及游戏竞争心理,拼手气类的活动甚至可以影响人们消费心理. 拼手气红包就是最简单的例子,哪怕你手气红包只有0.01元,在众多竞争者中脱颖而出,抢到的那一 ...

  4. 拼手气红包算法_二倍均值法

    使用二倍均值法进行的拼手气红包算法 假设M为总金额,N为抢红包人数,那么根据二倍均值法,每次抢到的金额 = 随机区间 (0, M / N X 2) 这个公式可以确保每个人获取的金额的平均值是相等的,不 ...

  5. java微信红包_Java模拟微信发红包(普通红包、拼手气红包)

    假设红包总额M元,分给N个人. 满足条件: 如果是普通红包,每个人获得的金额都一样: 如果是拼手气红包,则有所区别,但不能金额过于离谱,比如第一个获得的太多,以至于后来的人都几乎没得分. 注意点: 1 ...

  6. JAVA 拼手气红包 领取算法 记录

    废话不多说,直接上代码 //平分红包public static final Integer normalPacket = 1;//拼手气红包public static final Integer lu ...

  7. 拼手气红包java_JAVA实现拼手气红包算法

    实现拼手气红包算法,有以下几个需要注意的地方: 抢红包的期望收益应与先后顺序无关 保证每个用户至少能抢到一个预设的最小金额,人民币红包设置的最小金额一般是0.01元,如果需要发其他货币类型的红包,比如 ...

  8. h5拼手气红包java_Java模拟微信发红包(普通红包、拼手气红包)

    假设红包总额M元,分给N个人. 满足条件: 如果是普通红包,每个人获得的金额都一样: 如果是拼手气红包,则有所区别,但不能金额过于离谱,比如第一个获得的太多,以至于后来的人都几乎没得分. 注意点: 1 ...

  9. 微信拼手气红包算法(二倍平均值法)实现示范代码

    二倍平均值法 优势在于不需要先把每个人得多少算出来,只有拆红包的时候才会算出这个人得多少,减少实时算力 同时保证了每个人拼手气得到的平均值相同 设有10个人,红包总额100元. 100/10X2 = ...

最新文章

  1. python 非线性回归_机器学习入门之菜鸟之路——机器学习之非线性回归个人理解及python实现...
  2. 如何让AI教机器自己玩俄罗斯方块?
  3. 电商系统如何做搜索引擎?
  4. Spring boot 注解 ConfigurationProperties 的使用
  5. 同一局域网内_Pycharm访问服务器
  6. 春招,这 110 道 Python 面试题你看了吗?
  7. 【Docker】Error: No such image: gotok8s/kube-proxy:v1.16.5
  8. 中英文对照 —— 体育与健身
  9. Android 项目中用得最多最火的第三方框架可能都在这里了
  10. linux软件装在哪了,linux软件的安装目录在哪
  11. regexp用法mysql_MySQL中REGEXP正则表达式使用大全
  12. 百度UEditor编辑器压缩(缩放)图片只压缩jpg格式的解决方法
  13. 恒辉信达全数据AI管控云平台动态运维管控
  14. SecureCRT背景颜色
  15. DjangoUeditor添加与配置
  16. arcgis 空间交集 计算_ArcGIS叠置分析之相交分析
  17. linux清除历史linux清除history
  18. react-native使用高德地图获取当前地理位置
  19. 用户余额充值、提现操作。
  20. 正则表达式 之 /g /m /i 的区别

热门文章

  1. Atari游戏公司推出两款热门游戏的“区块链版本”
  2. 郭为重读麦肯锡报告:神州数码转型七年之痒
  3. 微信证据以及数据恢复删除记录恢复交易恢复
  4. 四十一 毕设 (上) 我在软件园的那些日子里
  5. 是什么让数据分析软件SAS风靡全球
  6. Spring boot Failed to bind properties under ‘XXX‘问题
  7. 独享云虚拟主机、共享云虚拟主机、云服务器 ECS 的区别
  8. java-php-python-ssm校园疫情防控管理系统计算机毕业设计
  9. WikiTaxi_Importer_1.3.1 维基 离线数据库
  10. 龙卷风迁徙地图,原来可以这样做