abstract:抽獎算法需要滿足的需求如下:1.可以控制中獎的概率2.具有隨機(jī)性3.最好可以控制獎品的數(shù)量4.根據(jù)用戶ID或者ip、手機(jī)號、QQ號等條件限制抽獎次數(shù)初期就這些需求,然后根據(jù)網(wǎng)上的資料,采用了一種階段式抽取的方法,大家下面看一下整體的程序:該程序是在ThinkPHP框架下完成的,使用了一些框架自帶的類庫和函數(shù),下面我會逐一進(jìn)行說明,控制器部分:代碼如下<?php /*
抽獎算法需要滿足的需求如下:
1.可以控制中獎的概率
2.具有隨機(jī)性
3.最好可以控制獎品的數(shù)量
4.根據(jù)用戶ID或者ip、手機(jī)號、QQ號等條件限制抽獎次數(shù)
初期就這些需求,然后根據(jù)網(wǎng)上的資料,采用了一種階段式抽取的方法,大家下面看一下整體的程序:
該程序是在ThinkPHP框架下完成的,使用了一些框架自帶的類庫和函數(shù),下面我會逐一進(jìn)行說明,控制器部分:
代碼如下
<?php /** * * * @lanfengye <zibin_5257@163.com> */ class ChoujiangAction extends Action { //抽獎的開始時間 var $begin_time="2012-12-25 14:00:00"; //開始時間 0-不限制 //抽獎的結(jié)束時間 var $stop_time="0"; //結(jié)束時間 0-不限制 //本次抽獎的獎項(xiàng)信息,必須按照從大到小的順序進(jìn)行填寫,id為獎次,prize為中獎信息,v為中獎概率,num為獎品數(shù)量 //需要注意的是,該處也必須包含不中獎的信息,概率從小到大進(jìn)行排序 var $prize_arr = array( '0' => array('id' => 1, 'prize' => '44元購買1G/年空間', 'v' => 1,'num'=>1), '1' => array('id' => 2, 'prize' => '55元購買1G/年空間', 'v' => 2,'num'=>2), '2' => array('id' => 3, 'prize' => '66元購買1G/年空間', 'v' => 5,'num'=>2), '3' => array('id' => 4, 'prize' => '77元購買1G/年空間', 'v' => 10,'num'=>3), '4' => array('id' => 5, 'prize' => '88元購買1G/年空間', 'v' => 15,'num'=>4), '5' => array('id' => 6, 'prize' => '99元購買1G/年空間', 'v' => 67,'num'=>10), ); //首頁顯示方法 public function index(){ //連接數(shù)據(jù)庫,去獲取本次中獎的人員名單 $Choujiang=M('Choujiang'); $this->assign('list', $Choujiang->where("rid>0")->order('id desc')->SELECT()); unset($Choujiang); //在首頁中顯示抽獎的開始時間 $this->assign('begin_time',$this->begin_time); $this->display(); } /** * 生成中獎信息,ajax進(jìn)行請求該方法,需要客戶填寫QQ號碼 */ public function make() { $qq_no= trim($_POST['qq_no']); import('ORG.Util.Input'); $qq_no=Input::getVar($qq_no); if(empty($qq_no)){ $this->ajaxReturn(1, '請正確填寫QQ號碼!'); exit; } if(!empty($this->begin_time) && time()<strtotime($this->begin_time)){ $this->ajaxReturn(1, '抽獎還沒有開始,開始時間為:'.$this->begin_time); exit; } if(!empty($this->stop_time) && time()>strtotime($this->stop_time)){ $this->ajaxReturn(1, '本次抽獎已經(jīng)結(jié)束,結(jié)束時間為:'.$this->stop_time); exit; } //獲取獎項(xiàng)信息數(shù)組,來源于私有成員 $prize_arr= $this->prize_arr; foreach ($prize_arr as $key => $val) { $arr[$val['id']] = $val['v']; } //$rid中獎的序列號碼 $rid = $this->get_rand($arr); //根據(jù)概率獲取獎項(xiàng)id $str = $prize_arr[$rid - 1]['prize']; //中獎項(xiàng) $Choujiang=M('Choujiang'); //從數(shù)據(jù)庫中獲取特定QQ號已經(jīng)參加抽獎的次數(shù),如果大于等于3則提示次數(shù)用完 if($Choujiang->where("qq_no='{$qq_no}'")->count()>=3){ $str='您3次抽獎機(jī)會已經(jīng)用完!'; $rid=0; //從數(shù)據(jù)庫中獲取特定獎項(xiàng)序號的次數(shù),大于等于設(shè)置的最大次數(shù)則提示獎品被抽完,如果需要一直中最后一個紀(jì)念獎,則修改該處即可 }elseif ($Choujiang->where("rid={$rid}")->count()>=$prize_arr[$rid-1]['num']) { $str='很抱歉,您所抽中的獎項(xiàng)已經(jīng)中完!'; $rid=0; } //生成一個用戶抽獎的數(shù)據(jù),用來記錄到數(shù)據(jù)庫 $data=array( 'rid'=>$rid, 'pop'=>$str, 'qq_no'=>$qq_no, 'input_time'=>time() ); //將用戶抽獎信息數(shù)組寫入數(shù)據(jù)庫 $Choujiang->add($data); unset($Choujiang); //ajax返回信息 $this->ajaxReturn(1, $str); } /** * 根據(jù)概率獲取中獎號碼 */ private function get_rand($proArr) { $result = ''; //概率數(shù)組的總概率精度 $proSum = array_sum($proArr); //概率數(shù)組循環(huán) foreach ($proArr as $key => $proCur) { $randNum = mt_rand(1, $proSum); if ($randNum <= $proCur) { $result = $key; break; } else { $proSum -= $proCur; } } unset($proArr); return $result; } } ?>
該算法簡單使用,并發(fā)訪問性能非常好,稍加改動就可以用于各種場合,結(jié)合用戶登錄等信息可有效控制每個人的抽獎次數(shù)。將開始和結(jié)束之間更改為數(shù)組,就可以完善成為每天特定時間抽獎的程序。
更多關(guān)于PHP抽獎算法程序代碼請關(guān)注PHP中文網(wǎng)(m.miracleart.cn)其他文章!