抽奖算法需要满足的需求如下:
1.可以控制中奖的概率
2.具有随机性
3.最好可以控制奖品的数量
4.根据用户ID或者ip、手机号、QQ号等条件限制抽奖次数
初期就这些需求,然后根据网上的资料,采用了一种阶段式抽取的方法,大家下面看一下整体的程序:
该程序是在ThinkPHP框架下完成的,使用了一些框架自带的类库和函数,下面我会逐一进行说明,控制器部分:
代码如下
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
|
<?php /** * * * @lanfengye <zibin_5257@163.com> */ class ChoujiangAction extends Action { //抽奖的开始时间 var $begin_time = "2012-12-25 14:00:00" ; //开始时间 0-不限制 //抽奖的结束时间 var $stop_time = "0" ; //结束时间 0-不限制 //本次抽奖的奖项信息,必须按照从大到小的顺序进行填写,id为奖次,prize为中奖信息,v为中奖概率,num为奖品数量 //需要注意的是,该处也必须包含不中奖的信息,概率从小到大进行排序 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(){ //连接数据库,去获取本次中奖的人员名单 $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进行请求该方法,需要客户填写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, '本次抽奖已经结束,结束时间为:' . $this ->stop_time); exit ; } //获取奖项信息数组,来源于私有成员 $prize_arr = $this ->prize_arr; foreach ( $prize_arr as $key => $val ) { $arr [ $val [ 'id' ]] = $val [ 'v' ]; } //$rid中奖的序列号码 $rid = $this ->get_rand( $arr ); //根据概率获取奖项id $str = $prize_arr [ $rid - 1][ 'prize' ]; //中奖项 $Choujiang =M( 'Choujiang' ); //从数据库中获取特定QQ号已经参加抽奖的次数,如果大于等于3则提示次数用完 if ( $Choujiang ->where( "qq_no='{$qq_no}'" )-> count ()>=3){ $str = '您3次抽奖机会已经用完!' ; $rid =0; //从数据库中获取特定奖项序号的次数,大于等于设置的最大次数则提示奖品被抽完,如果需要一直中最后一个纪念奖,则修改该处即可 } elseif ( $Choujiang ->where( "rid={$rid}" )-> count ()>= $prize_arr [ $rid -1][ 'num' ]) { $str = '很抱歉,您所抽中的奖项已经中完!' ; $rid =0; } //生成一个用户抽奖的数据,用来记录到数据库 $data = array ( 'rid' => $rid , 'pop' => $str , 'qq_no' => $qq_no , 'input_time' =>time() ); //将用户抽奖信息数组写入数据库 $Choujiang ->add( $data ); unset( $Choujiang ); //ajax返回信息 $this ->ajaxReturn(1, $str ); } /** * 根据概率获取中奖号码 */ private 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 ; } } ?> |
该算法简单使用,并发访问性能非常好,稍加改动就可以用于各种场合,结合用户登录等信息可有效控制每个人的抽奖次数。将开始和结束之间更改为数组,就可以完善成为每天特定时间抽奖的程序。
希望大家可以结合有关php抽奖程序的文章进行深入学习,更好的掌握这门语言的应用技巧。