抢红包算法

时间:2020-08-12 17:18:19

抢红包大家都知道,但发出一个固定金额的红包,由若干个人来抢,需要满足哪些规则?

  1. 所有人抢到金额之和等于红包金额,不能超过,也不能少于。

  2. 每个人至少抢到一分钱。

  3. 要保证所有人抢到金额的几率相等。

下面实现了两种抢红包的方法:二倍均值法 和 线段切割法

1、二倍均值法

设剩余红包金额为M,剩余人数为N,那么有如下公式:每次抢到的金额 = Random(0, M / N * 2)

这个公式,保证了每次随机金额的平均值是相等的,不会因为抢红包的先后顺序而造成不公平。

举个栗子:

假设有10个人,红包总额100元。

100 / 10 * 2 = 20,所以第一个人的随机范围是(0,20),平均可以抢到 10 元。假设第一个人随机到 10 元,那么剩余金额是100 - 10 = 90 元。

90 / 9 * 2 = 20,所以第二个人的随机范围同样是(0,20 ),平均可以抢到 10 元。假设第二个人随机到10元,那么剩余金额是90-10 = 80 元。

80/8X2 = 20, 所以第三个人的随机范围同样是(0,20 ),平均可以抢到10元。

以此类推,除了最后一次,每一次随机范围的均值是相等的。

public function divideRedPackage()
{
    $totalAmount = 100; // 元
    $totalPeopleNum = 10;// 人
    $totalBalance = $totalAmount * 100;// 转成分
    $arrOneAmount = [];// 结果
    for ($i = 0; $i < $totalPeopleNum - 1; $i++) {
        //设剩余红包金额为M,剩余人数为N,那么有如下公式:每次抢到的金额 = Random(0, M / N * 2)
        $tmp = random_int(0, $totalBalance / ($totalPeopleNum - $i) * 2) + 1;
        $totalBalance -= $tmp;
        $arrOneAmount[$i] = $tmp / 100;
    }
    // 最后一次单独处理
    $arrOneAmount[++$i] = $totalBalance / 100;
    print_r($arrOneAmount);
    echo '确认和为:' . array_sum($arrOneAmount);
}

图片.png

图片.png

2、线段切割法

何谓线段切割法?我们可以把红包总金额想象成一条很长的线段,而每个人抢到的金额,则是这条主线段所拆分出的若干子线段。

如何确定每一条子线段的长度呢?由“切割点”来决定。当 N 个人一起抢红包的时候,就需要确定 N-1 个切割点。因此,我们需要做 N-1 次随机运算,以此确定 N-1 个切割点。随机的范围区间是(1, M)。

当所有切割点确定以后,子线段的长度也随之确定。这样每个人来抢红包的时候,只需要顺次领取与子线段长度等价的红包金额即可。

这就是线段切割法的思路。在这里需要注意以下两点:

  1. 当随机切割点出现重复,如何处理。

  2. 如何尽可能降低时间复杂度和空间复杂度。 

public function lineCut()
{
    $totalAmount = 100; // 元
    $totalPeopleNum = 10;// 人
    $totalBalance = $totalAmount * 100;// 转成分
    $arrOneAmount = [];// 结果
    // 确认切隔点
    $j = 1;
    $arrRand = [];
    while ($j < $totalPeopleNum) {
        $tmp = mt_rand(1, $totalBalance - 1);
        if (!in_array($tmp, $arrRand)) {
            $arrRand[$j] = $tmp;
            $j++;
        }
    }
    sort($arrRand);
    print_r($arrRand);// 分隔的点
    foreach ($arrRand as $key => $one) {
        $arrOneAmount[] = $key == 0 ? $one / 100 : ($one - $arrRand[$key - 1]) / 100;
    }
    // 最后一个
    $arrOneAmount[] = ($totalBalance - array_pop($arrRand)) / 100;
    print_r($arrOneAmount);
    echo '确认和为:' . array_sum($arrOneAmount);
}

图片.png

漫画:如何实现抢红包算法?

  • 9
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值