php实现概率性随机抽奖代码


Posted in PHP onJanuary 02, 2016

1、初始数据:

权重越大,抽取的几率越高
[奖品1, 权重 5], [ 奖品2, 权重6], [ 奖品3, 权重 7], [ 奖品4, 权重2]

2、处理步骤:

1)N = 5 + 6 + 7 + 2 = 20
2)然后取1-N的随机数M
3)界定各 奖品的权重范围值 奖品 1 : 1-5 ; 奖品2 : 6-11; 奖品3: 12-18; 奖品4: 19-20
4) 如果M在某个奖品的权重范围值内,标识这个奖品被抽取到

<?php
/**
 * 奖品
 */
class Prize {
  # ID
  public $id = null;
  # 权重
  public $weight = null;
  # 奖品名
  public $name = null;
 
  # 权重范围区间起始值
  protected $start = 0;
  # 权重范围区间结束值
  protected $end = 0;
 
  public function __construct($id, $weight, $name) {
    if (!$id) {
      throw new Exception('奖品ID为空.');
    }
    $this->id = $id;
    $this->weight = $weight ? $weight : 0;
    $this->name = $name ? $name : '随机奖品' . $id;
  }
 
  # id
  public function getId() {
    return $this->id;
  }
 
  # 权重
  public function getWeight() {
    return $this->weight;
  }
 
  # 设置权重范围区间
  public function setRange($start, $end) {
    $this->start = $start;
    $this->end = $end;
  }
 
  # 判断随机数是否在权重范围区间
  public function inRange($num) {
    return ($num >= $this->start) && ($num <= $this->end);
  }
}
 
/**
 * 奖品池
 */
class PrizePoll implements IteratorAggregate, Countable {
  # 奖品集
  protected $items = array();
 
  # 加入奖品
  public function addItem(Prize $item) {
    $this->items[$item->getId()] = $item;
    return $this;
  }
 
  # 删除奖品
  public function removeItem($itemId) {
    if (isset($this->items[$itemId])) {
      unset($this->items[$itemId]);
    }
    return $this;
  }
 
  # 更新奖品
  public function updateItem(Prize $item) {
    if (isset($this->items[$item->getId()])) {
      $this->items[$item->getId()] = $item;
    }
    return $this;
  }
 
  # 获取所有奖品
  public function getItems() {
    return $this->items;
  }
 
  # 所有所有可用奖品(如果权重为0,说明这个奖品永远不可能抽到)
  public function getVisibleItems() {
    $items = array();
    foreach ($this->items as $item) {
      if ($item->getWeight()) {
        $items[$item->getId()] = $item;
      }
    }
    return $items;
  }
 
  # Countable::count
  public function count() {
    return count($this->items);
  }
 
  # IteratorAggregate::getIterator()
  public function getIterator() {
    return new ArrayIterator($this->items);
  }
}
 
/**
 * 简单的抽奖类
 */
class SimpleTurn {
  # 奖池
  protected $poll = null;
   
  public function __construct(PrizePoll $poll) {
    if ($poll) {
      $this->setPoll($poll);
    }
  }
 
  # 抽奖
  public function run(PrizePoll $poll) {
    $poll = $poll ? $poll : $this->poll;
    if ( ! $poll) {
      throw new Exception('奖池未初始化');
    }
 
    if ($poll->count() <= 0) {
      throw new Exception('奖池为空');
    }
 
    $items = $poll->getVisibleItems();
    if (count($items) <= 0) {
      throw new Exception('奖池为空');
    }
 
    $sum = 0;
    foreach ($items as $item) {
      $start = $sum + 1;
      $sum += $item->getWeight();
      $end = $sum;
 
      # 设置奖品的权重范围区间
      $item->setRange($start, $end);
    }
 
    # 随机数
    $rand = $this->getRandNum(1, $sum);
 
    # 区间段判断
    foreach ($items as $item) {
      if ($item->inRange($rand)) {
        return $item;
      }
    }
    return null;
  }
 
  # 获取随机数
  public function getRandNum($min, $max) {
    return mt_rand($min ? $min : 1, $max);
  }
 
  # 设置奖池
  public function setPoll(PrizePoll $poll) {
    $this->poll = $poll;
  }
}
 
# 示例
try {
  $prizePoll = new PrizePoll();
  $prizePoll->addItem(new Prize(1, 5))
    ->addItem(new Prize(2, 6))
    ->addItem(new Prize(3, 7))
    ->addItem(new Prize(4, 2));
 
  $turn = new SimpleTurn($prizePoll);
  $prize = $turn->run();
  var_dump($prize);
} catch (Exception $e) {
  print_r($e);
}
PHP 相关文章推荐
IIS下配置Php+Mysql+zend的图文教程
Dec 08 PHP
隐藏X-Space个人空间下方版权方法隐藏X-Space个人空间标题隐藏X-Space个人空间管理版权方法
Feb 22 PHP
动易数据转成dedecms的php程序
Apr 07 PHP
需要注意的几个PHP漏洞小结
Feb 05 PHP
thinkphp3.0 模板中函数的使用
Nov 13 PHP
PHP删除非空目录的函数代码小结
Feb 28 PHP
PHP闭包(Closure)使用详解
May 02 PHP
PHP return语句的另一个作用
Jul 30 PHP
php获取远程文件的内容和大小
Nov 03 PHP
Thinkphp框架开发移动端接口(2)
Aug 18 PHP
Laravel 自动生成验证的实例讲解:login / logout
Oct 14 PHP
PHP实现常用排序算法的方法
Feb 05 PHP
php实现给一张图片加上水印效果
Jan 02 #PHP
前端必学之PHP语法基础
Jan 01 #PHP
WordPress中登陆后关闭登陆页面及设置用户不可见栏目
Dec 31 #PHP
WordPress中获取指定分类及其子分类下的文章数目
Dec 31 #PHP
Swoole-1.7.22 版本已发布,修复PHP7相关问题
Dec 31 #PHP
WordPress中给媒体文件添加分类和标签的PHP功能实现
Dec 31 #PHP
简单了解将WordPress中的工具栏移到底部的小技巧
Dec 31 #PHP
You might like
德生1994机评
2021/03/02 无线电
加速XP搜索功能堪比vista
2007/03/22 PHP
解析PHP多种序列化与反序列化的方法
2013/06/06 PHP
destoon复制新模块的方法
2014/06/21 PHP
php文件夹的创建与删除方法
2015/01/24 PHP
YII CLinkPager分页类扩展增加显示共多少页
2016/01/29 PHP
取消选中单选框radio的三种方式示例介绍
2013/12/23 Javascript
jquery 取子节点及当前节点属性值
2014/07/25 Javascript
AngularJS实现表单验证
2015/01/28 Javascript
原生js实现的贪吃蛇网页版游戏完整实例
2015/05/18 Javascript
如何用angularjs制作一个完整的表格
2016/01/21 Javascript
node.js实现端口转发
2016/04/14 Javascript
BootStrap文件上传样式超好看【持续更新】
2016/05/10 Javascript
DOM操作和jQuery实现选项移动操作的简单实例
2016/06/07 Javascript
html中鼠标滚轮事件onmousewheel的处理方法
2016/11/11 Javascript
JavaScript微信定位功能实现方法
2016/11/29 Javascript
selenium 与 chrome 进行qq登录并发邮件操作实例详解
2017/04/06 Javascript
jQuery Datatable 多个查询条件自定义提交事件(推荐)
2017/08/24 jQuery
一文总结学习Python的14张思维导图
2017/10/17 Python
Python中函数参数调用方式分析
2018/08/09 Python
使用Python获取并处理IP的类型及格式方法
2018/11/01 Python
解决Python2.7中IDLE启动没有反应的问题
2018/11/30 Python
Python基本socket通信控制操作示例
2019/01/30 Python
在keras中获取某一层上的feature map实例
2020/01/24 Python
Python安装与卸载流程详细步骤(图解)
2020/02/20 Python
canvas粒子动画背景的实现示例
2018/09/03 HTML / CSS
Linux机考试题
2015/10/16 面试题
年终考核评语
2014/01/19 职场文书
八荣八耻演讲稿
2014/09/15 职场文书
教师考核评语大全
2014/12/31 职场文书
试用期自我评价怎么写
2015/03/10 职场文书
2015年中秋寄语
2015/07/31 职场文书
AI:如何训练机器学习的模型
2021/04/16 Python
详解Node.js如何处理ES6模块
2021/05/15 Javascript
Python快速优雅的批量修改Word文档样式
2021/05/20 Python
每日六道java新手入门面试题,通往自由的道路
2021/06/30 Java/Android