抽奖概率算法
流程是通过计算概率的和,再生成处于这个范围的随机数。循环判断随机数所处的哪个范围,如果找到就返回当前范围的key,如果没找到就减去当前范围,继续循环找。最后通过key找到对应的奖项。
<?php /** * 生成抽奖抽中项 * @param $proArr * @return int|string */ function getRand($proArr) { $result = ''; //概率数组的总概率精度 $proSum = array_sum($proArr); //概率数组循环 foreach ($proArr as $key => $proCur) { // 生成范围内随机值 $randNum = mt_rand(1, $proSum); // 判断是否在当前范围内,如果是记录key,如果不是则在总概率中减去当前范围,并继续判断下一个 if ($randNum <= $proCur) { $result = $key; break; } else { // (缩小范围) $proSum -= $proCur; } } unset ($proArr); // 返回key return $result; } // 奖项和概率(可以保存到redis的hash中,如果被抽中把除未中奖外的概率置0) $prizeArr = [ ['id' => 1, 'prize' => '一等奖', 'v' => 1], ['id' => 2, 'prize' => '二等奖', 'v' => 5], ['id' => 3, 'prize' => '三等奖', 'v' => 10], ['id' => 4, 'prize' => '四等奖', 'v' => 12], ['id' => 5, 'prize' => '五等奖', 'v' => 22], ['id' => 6, 'prize' => '未中奖', 'v' => 50], ]; // 生成id和奖项映射 foreach ($prizeArr as $key => $val) { $arr[$val['id']] = $val['v']; } // 获取抽中的key $rid = getRand($arr); // 获取结果 echo $prizeArr[$rid - 1]['prize'];