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 相关文章推荐
Extended CHM PHP 语法手册之 DIY
Oct 09 PHP
一个PHP日历程序
Dec 06 PHP
php中定义网站根目录的常用方法
Aug 08 PHP
php class中self,parent,this的区别以及实例介绍
Apr 24 PHP
详解PHP内置访问资源的超时时间 time_out file_get_contents read_file
Jun 03 PHP
CI框架安全类Security.php源码分析
Nov 04 PHP
php判断数组中是否存在指定键(key)的方法
Mar 17 PHP
几个优化WordPress中JavaScript加载体验的插件介绍
Dec 17 PHP
Yii使用技巧大汇总
Dec 29 PHP
PHP+Ajax实现的无刷新分页功能详解【附demo源码下载】
Jul 03 PHP
如何通过View::first使用Laravel Blade的动态模板详解
Sep 21 PHP
Laravel 加载第三方类库的方法
Apr 20 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
PHP中模拟处理HTTP PUT请求的例子
2014/07/22 PHP
PHP实现模仿socket请求返回页面的方法
2014/11/04 PHP
javascript数组操作(创建、元素删除、数组的拷贝)
2014/04/07 Javascript
JavaScript实现添加及删除事件的方法小结
2015/08/04 Javascript
jQuery-1.9.1源码分析系列(十一)DOM操作续之克隆节点
2015/12/01 Javascript
谈谈我对JavaScript原型和闭包系列理解(随手笔记6)
2015/12/20 Javascript
Javascript实现汉字和拼音互转的终极方案
2016/10/19 Javascript
node.js版本管理工具n无效的原理和解决方法
2016/11/24 Javascript
js实现3d悬浮效果
2017/02/16 Javascript
ES6中新增的Object.assign()方法详解
2017/09/22 Javascript
create-react-app构建项目慢的解决方法
2018/03/14 Javascript
基于Vue自定义指令实现按钮级权限控制思路详解
2018/05/23 Javascript
详解js创建对象的几种方法及继承
2019/04/12 Javascript
详解利用nodejs对本地json文件进行增删改查
2019/09/20 NodeJs
vue - vue.config.js中devServer配置方式
2019/10/30 Javascript
js实现贪吃蛇游戏 canvas绘制地图
2020/09/09 Javascript
[58:21]DOTA2亚洲邀请赛 4.3 突围赛 Liquid vs VGJ.T 第二场
2018/04/04 DOTA
python绘图方法实例入门
2015/05/19 Python
详解使用Python处理文件目录的相关方法
2015/10/16 Python
python实现解数独程序代码
2017/04/12 Python
Python操作MySQL数据库的三种方法总结
2018/01/30 Python
Python实现在某个数组中查找一个值的算法示例
2018/06/27 Python
使用tensorflow实现线性回归
2018/09/08 Python
python实现淘宝秒杀脚本
2020/06/23 Python
python处理DICOM并计算三维模型体积
2019/02/26 Python
Python生成器的使用方法和示例代码
2019/03/04 Python
Python异常处理例题整理
2019/07/07 Python
python3发送邮件需要经过代理服务器的示例代码
2019/07/25 Python
django删除表重建的实现方法
2019/08/28 Python
PyCharm2020.3.2安装超详细教程
2021/02/08 Python
Django视图类型总结
2021/02/17 Python
MAC Cosmetics官方网站:魅可专业艺术彩妆
2019/04/10 全球购物
2019毕业典礼主持词!
2019/07/05 职场文书
七年级作文之游记
2019/12/11 职场文书
HTML常用标签超详细整理
2022/03/19 HTML / CSS
js 实现Material UI点击涟漪效果示例
2022/09/23 Javascript