php经典抽奖实现,刮刮乐、大转盘等功能经典算法

抽奖是非常重要的一个和粉丝互动的部分,在互动的时候可以引入更多的营销手段。

下面展示的是经典的php实现抽奖逻辑一个简单算法,但是非常实用,对于一些小型的活动可以直接修改一下用得很好。

<?php
/**
 * Created by Brisklan.
 * User: 1076963452@qq.com
 * Date: 2018/12/23 11:05
 * 经典的概率算法,
 * $proArr是一个预先设置的数组,
 * 假设数组为:array ( 1 => 1, 2 => 5, 3 => 10, 4 => 12, 5 => 22, 6 => 50 ),
 * 开始是从1,1000 这个概率范围内筛选第一个数是否在他的出现概率范围之内,
 * 如果不在,则将概率空间,也就是k的值减去刚刚的那个数字的概率空间,
 * 在本例当中就是减去100,也就是说第二个数是在1,900这个范围内筛选的。
 * 这样 筛选到最终,总会有一个数满足要求。
 * 就相当于去一个箱子里摸东西,
 * 第一个不是,第二个不是,第三个还不是,那最后一个一定是。
 * 这个算法简单,而且效率非常高,
 * 这个算法在大数据量的项目中效率非常棒。
 */
function get_rand($proArr)
{
    $result = '';
    //概率数组的总概率精度
    $proSum = array_sum($proArr);
    //概率数组循环
    foreach ($proArr as $key => $proCur) {
        $randNum = mt_rand(1, $proSum);
        if ($randNum <= $proCur) {
            $result = $key;
            break;
        } else {
            $proSum -= $proCur;
        }
    }
    unset ($proArr);
    return $result;
}

/*
 * 奖项数组
 * 是一个二维数组,记录了所有本次抽奖的奖项信息,
 * 其中id表示中奖等级,prize表示奖品,v表示中奖概率。
 * 注意其中的v必须为整数,你可以将对应的 奖项的v设置成0,即意味着该奖项抽中的几率是0,
 * 数组中v的总和(基数),基数越大越能体现概率的准确性。
 * 本例中v的总和为1000,那么免费服务器对应的中奖概率就是千分之一,免费数据库的中奖概率是千分之19,
 * 如果v的总和是10000,那中奖概率就是万分之一了。
 */
function getPrizes()
{
    return array(
        '0' => array('id' => 1, 'prize' => '免费服务器', 'v' => 1),
        '1' => array('id' => 2, 'prize' => '免费数据库', 'v' => 19),
        '2' => array('id' => 3, 'prize' => '免费公众号官网', 'v' => 100),
        '3' => array('id' => 4, 'prize' => '免费服装版小程序', 'v' => 100),
        '4' => array('id' => 5, 'prize' => '免费生鲜版小程序', 'v' => 150),
        '5' => array('id' => 6, 'prize' => '免费小商户版小程序', 'v' => 100),
        '6' => array('id' => 7, 'prize' => '1年免费物联网自动化系统', 'v' => 80),
        '7' => array('id' => 8, 'prize' => '免费物流发货系统', 'v' => 50),
        '8' => array('id' => 9, 'prize' => '哎呀,与奖品擦肩而过了,继续努力哦', 'v' => 400),
    );
}

/*
 * 每次前端页面的请求,PHP循环奖项设置数组,
 * 通过概率计算函数get_rand获取抽中的奖项id。
 * 将中奖奖品保存在数组$res['yes']中,
 * 而剩下的未中奖的信息保存在$res['no']中,
 * 最后输出json个数数据给前端页面。
 * 2021.04.07 优化了奖品二维数组排序问题,新增返回中奖数组功能,方便做其他处理
 */
function getResult($prize_arr,$retYesArr=false)
{
    $arr = [];
    foreach ($prize_arr as $key => $val) {
        $arr[$val['id']] = $val['v'];
    }

    $rid = $this->get_rand($arr); //根据概率获取奖项id

    //用ID作为数组key重组多维数组
    $temArr = array_column($prize_arr,null,'id');
    //只返回中奖的项目
    if ($retYesArr){
        return $temArr[$rid];
    }

    $res['yes'] = $temArr[$rid]['prize']; //中奖项
    unset($temArr[$rid]); //将中奖项从数组中剔除,剩下未中奖项
    shuffle($temArr); //打乱数组顺序
    $pr = [];
    $count = count($temArr);
    for ($i = 0; $i < $count; $i++) {
        $pr[] = $temArr[$i]['prize'];
    }
    $res['no'] = $pr;
    return $res;
}
echo '<pre>';
print_r(getResult(getPrizes(),false));



下面的测试的结果:

Array
(
    [yes] => 免费生鲜版小程序
    [no] => Array
        (
            [0] => 免费物流发货系统
            [1] => 免费数据库
            [2] => 免费公众号官网
            [3] => 免费小商户版小程序
            [4] => 免费服务器
            [5] => 1年免费物联网自动化系统
            [6] => 免费服装版小程序
            [7] => 哎呀,与奖品擦肩而过了,继续努力哦
        )

)

将方法简单调整就可以做出一个系统的插件提供给客户选择了。


评论/留言