红包算法之二倍均值算法-php、js版示例代码

2023-06-01 00:00:00 算法 红包 均值

二倍均值法,PHP版,js版实现示例代码


php版:

/**
 * 红包
 * @param integer $money 红包金额。单位:分
 * @param integer $p_num 人数
 * @param integer $min_money 每个红包最小金额。单位:分
 * @param integer $multiple 倍数。值越大,红包差额越大,也就是最后的红包都是最小值
 * @return data[integer]
 */
function red_packet(int $money, int $p_num, int $min_money = 1, int $multiple = 2)
{
    $red_packets = [];
    for ($i = $p_num; $i > 0; $i--) {
        if ($i == 1) {
            $red_packets[] = $money;
        } else {
            $rand_money = random_int($min_money, floor($money / $p_num * $multiple));
            $p_num--;
            if ($p_num * $min_money > $money - $rand_money) {
                $rand_money -= $p_num * $min_money - ($money - $rand_money);
            }
            $money -= $rand_money;
            $red_packets[] = $rand_money;
        }
    }
    return $red_packets;
}

//ps:人数 × 最小金额 <= 红包金额

//调用测试
red_packet(100, 10); 


js版

/**
 * 获取指定区间内的随机整数,区间:[a, b)
 * @param a 区间下限(包含)
 * @param b 区间上限(不包含)
 * @return 一个随机整数
 */
function getRandomBetween(a, b) {
    return Math.floor(a + Math.random() * (b - a));
}
/**
 * 模拟抢红包,使用二倍均值法
 * @param money 总金额,单位:分
 * @param person 抢红包人数
 * @return 生成的红包金额数组
 */
function RedPacket(money, person) {
    var amountArr = new Array(person);
    console.log(`${money} 分钱 分给 ${person} 人`);
    for (var i = 0; i < amountArr.length - 1; i++) {
        var avgAmount = money / person;
        var doubleAvgAmount = avgAmount * 2;
        person--;
        var min = 1;
        var max = doubleAvgAmount;
        var currentAmount = getRandomBetween(min, max);
        amountArr[i] = currentAmount;
        money = money - currentAmount;
        console.log(`剩余人数:${person}\t抢到:${currentAmount} \t剩余金额:${money}\t本次均值的二倍:${doubleAvgAmount}\t金额随机范围:[${min}, ${max}]`);
    }
    amountArr[amountArr.length - 1] = money;
    return amountArr;
}

//ps:可直接复制到浏览器console运行测试

//调用测试
RedPacket(100, 10);

相关文章