本文实例讲述了php实现的微信红包算法。红包类型有两种:
1. 普通红包
2. 拼手气红包
普通红包就不用多解析了,大锅饭原理,平分。
拼手气红包讲的是手气(运气),有人可以抢到很多,有人抢的少得可怜,当然也不是先抢就一定多,说到底了就是随机。
// $bonus_total 红包总金额
// $bonus_count 红包个数
// $bonus_type 红包类型 1=拼手气红包 0=普通红包
function randBonus($bonus_total=0, $bonus_count=3, $bonus_type=1){
$bonus_items = array(); // 将要瓜分的结果
$bonus_balance = $bonus_total; // 每次分完之后的余额
$bonus_avg = number_format($bonus_total/$bonus_count, 2); // 平均每个红包多少钱
$i = 0;
while($i<$bonus_count){
if($i<$bonus_count-1){
$rand = $bonus_type?(rand(1, $bonus_balance*100-1)/100):$bonus_avg; // 根据红包 类型计算当前红包的金额
$bonus_items[] = $rand;
$bonus_balance -= $rand;
}else{
$bonus_items[] = $bonus_balance; // 最后一个红包直接承包最后所有的金额,保证发出的总金额正确
}
$i++;
}
return $bonus_items;
}
另一个使用数组实现的版本,原理差不多:
function sendRandBonus($total=0, $count=3, $type=1){
if($type==1){
$input = range(0.01, $total, 0.01);
if($count>1){
$rand_keys = (array) array_rand($input, $count-1);
$last = 0;
foreach($rand_keys as $i=>$key){
$current = $input[$key]-$last;
$items[] = $current;
$last = $input[$key];
}
}
$items[] = $total-array_sum($items);
}else{
$avg = number_format($total/$count, 2);
$i = 0;
while($i<$count){
$items[] = $i<$count-1?$avg:($total-array_sum($items));
$i++;
}
}
return $items;
}
/**
* 红包生成算法
* 自行去校验 $amount 与 $num 的值是否有效。
*
* @ApiParams $amount int 必须 类型 红包金额 单位为分
* @ApiParams $num int 必须 类型 红包数量
*/
function splitHongbao($amount, $num)
{
$red = []; //金额
$rate = [];// 占比
$allRate = 0; // 总值
$min = 0; //占比最小的索引
$allRed = 0; //二次填充后的总金额
//一次遍历
for ($i = 0; $i < $num; $i++) {
$red[$i] = 1; //分配低保
$rate[$i] = mt_rand(0, $num * 1000); //生成随机占比
$allRate = $rate[$i] + $allRate;//统计总占比
if ($rate[$i] < $rate[$min]) {
$min = $i; //记录占比最小的索引
}
}
//剩下的钱拿来随机分
$rest = $amount - $num;
if ($rest < 0) {
throw new Exception('金额错误');
}
//二次遍历 填充金额
for ($i = 0; $i < $num; $i++) {
$red[$i] = $red[$i] + floor($rest * ($rate[$i] / $allRate)); //不保留小数
$allRed = $allRed + $red[$i]; //累计金额
}
$rest = $amount - $allRed; //剩下一点点 分配给刚才最小的
$red[$min] = $red[$min] + $rest;
//三次遍历 校验红包金额总和是否正确
$count = 0;
for ($i = 0; $i < $num; $i++) {
$count = $count + $red[$i];
}
//数组打乱
shuffle($red);
return $red;
}