不同概率的抽奖

今天为大家写个小程序。

工作中有遇到一些抽奖的活动,但是你懂得,抽奖物品的概率肯定不是一样,你会发现好的东西很难抽到,经常抽到一些垃圾的东西,嘿嘿,这就是本文要说的,我们要控制抽奖物品的概率。还有顺便说一句,网上这种小程序几乎没有,很多都是等概率的抽奖balabala…

需求很简单,为了更加形象,这里我们列一个表格来显示我们抽奖的物品和对应的概率(没有边框,大家凑合着看看吧,不想改造Octopress的样式了)

序号 物品名称 物品ID 抽奖概率
1 物品1 P1 0.2
2 物品2 P2 0.1
3 物品3 P3 0.4
4 物品4 P4 0.3
5 物品5 P5 0.0
6 物品6 P6 -0.1
7 物品7 P7 0.008

数据很简单,那么就直接看代码了

VO类,具体对应就是上面表格里的内容

(Gift.java)download

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
packageorg.usc.usc.lottery;

publicclassGift {
privateintindex;
privateStringgitfId;
privateStringgiftName;
privatedoubleprobability;

publicGift(intindex, StringgitfId, StringgiftName, doubleprobability) {
this.index = index;
this.gitfId = gitfId;
this.giftName = giftName;
this.probability = probability;
    }

publicintgetIndex() {
returnindex;
    }

publicvoidsetIndex(intindex) {
this.index = index;
    }

publicStringgetGitfId() {
returngitfId;
    }

publicvoidsetGitfId(StringgitfId) {
this.gitfId = gitfId;
    }

publicStringgetGiftName() {
returngiftName;
    }

publicvoidsetGiftName(StringgiftName) {
this.giftName = giftName;
    }

publicdoublegetProbability() {
returnprobability;
    }

publicvoidsetProbability(doubleprobability) {
this.probability = probability;
    }

    @Override
publicStringtoString() {
return"Gift [index=" + index + ", gitfId=" + gitfId + ", giftName=" + giftName + ", probability=" + probability + "]";
    }
}

工具类,真正的不同概率的抽奖就在这里

(LotteryUtil.java)download

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
packageorg.usc.usc.lottery;

import java.util.ArrayList;
import java.util.Collections;
import java.util.List;

/**
 * 不同概率抽奖工具包
 *
 * @author Shunli
 */
publicclassLotteryUtil {
/**
     * 抽奖
     *
     * @param orignalRates
     *            原始的概率列表,保证顺序和实际物品对应
     * @return
     *         物品的索引
     */
publicstaticintlottery(List<Double> orignalRates) {
if (orignalRates == null || orignalRates.isEmpty()) {
return -1;
        }

intsize = orignalRates.size();

// 计算总概率,这样可以保证不一定总概率是1
doublesumRate = 0d;
for (doublerate : orignalRates) {
sumRate += rate;
        }

// 计算每个物品在总概率的基础下的概率情况
List<Double> sortOrignalRates = newArrayList<Double>(size);
DoubletempSumRate = 0d;
for (doublerate : orignalRates) {
tempSumRate += rate;
sortOrignalRates.add(tempSumRate / sumRate);
        }

// 根据区块值来获取抽取到的物品索引
doublenextDouble = Math.random();
sortOrignalRates.add(nextDouble);
Collections.sort(sortOrignalRates);

returnsortOrignalRates.indexOf(nextDouble);
    }
}

测试类,测试上面的抽奖是否成功,n次抽奖看抽奖结果

(LotteryTest.java)download

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
packageorg.usc.usc.lottery;

import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;

/**
 * 不同概率抽奖
 *
 * @author ShunLi
 */
publicclassLotteryTest {
publicstaticvoidmain(String[] args) {
List<Gift> gifts = newArrayList<Gift>();
// 序号==物品Id==物品名称==概率
gifts.add(newGift(1, "P1", "物品1", 0.2d));
gifts.add(newGift(2, "P2", "物品2", 0.2d));
gifts.add(newGift(3, "P3", "物品3", 0.4d));
gifts.add(newGift(4, "P4", "物品4", 0.3d));
gifts.add(newGift(5, "P5", "物品5", 0d));
gifts.add(newGift(6, "P6", "物品6", -0.1d));
gifts.add(newGift(7, "P7", "物品7", 0.008d));

List<Double> orignalRates = newArrayList<Double>(gifts.size());
for (Giftgift : gifts) {
doubleprobability = gift.getProbability();
if (probability < 0) {
probability = 0;
            }
orignalRates.add(probability);
        }

// // test
// for (int i = 0; i < 10000; i++) {
// try {
// Gift tuple = gifts.get(LotteryUtil.lottery(orignalRates));
// System.out.println(tuple);
// } catch (Exception e) {
// System.out.println("lottery failed, please check it!");
// }
// }

// statistics
Map<Integer, Integer> count = newHashMap<Integer, Integer>();
doublenum = 1000000;
for (inti = 0; i < num; i++) {
intorignalIndex = LotteryUtil.lottery(orignalRates);

Integervalue = count.get(orignalIndex);
count.put(orignalIndex, value == null ? 1 : value + 1);
        }

for (Entry<Integer, Integer> entry : count.entrySet()) {
System.out.println(gifts.get(entry.getKey()) + ", count=" + entry.getValue() + ", probability=" + entry.getValue() / num);
        }
    }
}

结果

1
2
3
4
5
Gift [index=1, gitfId=P1, giftName=物品1, probability=0.2], count=199139, probability=0.199139
Gift [index=2, gitfId=P2, giftName=物品2, probability=0.1], count=99328, probability=0.099328
Gift [index=3, gitfId=P3, giftName=物品3, probability=0.4], count=396575, probability=0.396575
Gift [index=4, gitfId=P4, giftName=物品4, probability=0.3], count=296997, probability=0.296997
Gift [index=7, gitfId=P7, giftName=物品7, probability=0.0080], count=7961, probability=0.007961

不同概率的抽奖原理很简单 
就是把0到1的区间分块,而分块的依据就是物品占整个的比重,再根据随机数种子来产生0-1中间的某个数,来判断这个数是落在哪个区间上,而对应的就是抽到了那个物品。随机数理论上是概率均等的,产生的每个数理论上也应该概率均等,那么相应的区间所含数的多少就体现了抽奖物品概率的不同。(p.s. 当然数目是数不清楚的,具体抽象话了点)

这个实例的数据可以说明 
1. 概率可以是负数和0,当然实际上中应该不会(p.s. 正常情况下可能真的有0,比如抽个iphone5,当然是抽不到的了,这个时候,构建礼物(List gifts)的时候最好就不要加这个进去),还有可以把负数的处理放到抽奖工具类(LotteryUtil)中; 
2. 所有礼物加起来的概率可以不是1,可以认为这里的概率是一个权重;

小小分享了,倒是觉得大家可以自己先想想,如果你来写这样的小程序,如何来写,有没有其它的创意和想法?如果有什么建议或问题的话,可以通过微博 http://weibo.com/lishunli 联系到我,大家一起交流学习。

时间: 2024-12-17 09:28:05

不同概率的抽奖的相关文章

C# 做一个指定概率的抽奖程序

static void Main(string[] args) { //各物品的概率保存在数组里 float[] area = new float[4]{ 0.980f, 0.550f, 0.230f, 0.010f }; //单次测试 //Console.WriteLine(Get(area)); //批量测试 int[] result = new int[4]{ 0, 0, 0, 0 }; for (int i = 0; i < 1770000; i++) //为了比对结果方便,这里循环的次

相同概率的抽奖程序另类实现——使用数据库,无需数学原理

抽奖,是很多企业.聚会的常见玩乐形式,光彩绚丽的抽奖屏幕背后,是计算程序+抽奖用户信息.程序=算法+数据结构. 好,说抽奖程序的的实现吧.这个实现一般需要应用数学原理.而本文的方法是我在参加一次婚礼的抽奖体验后突然想到的,一种比较简单.无需数学原理的方法. 功能:能按照相同概率,从用户集合中抽出随机的部分用户集合作为中奖者.抽奖可以进行多次,对已中奖的用户不会重复抽取. 使用技术: 1.SqlServer数据库,使用NewID()作为select随机筛选函数 2.sql随机函数 3.为了快速方便

编程题 - 概率问题,抽奖问题 48选7

编程题 : 有12组球,每组有编号为ABCD四个球. 任意球可记为 nX (1<=n<=12 ,X ={A B C D}).随机抽取7个球.问抽到结果中,存在7个不同组的球的可能性.6个呢? 进而(5,4,3,2个的可能性)? 解题思路:应用 编程题 -- 分组问题,输出公式(一) 中的结果,计算 grouping 7 7 4 ;grouping 7 6 4 ....得到: 7:7*16:5*1 + 1*2 ( 解释下这个结果,后面的就自然明了. 从5个组中各抽一个球且从另一个组中抽3个球 )

paip.刮刮卡砸金蛋抽奖概率算法跟核心流程.

#---抽奖算法需要满足的需求如下: 1 #---抽奖核心流程 1 #---问题???更好的算法 2 #---实际使用的扩展抽奖算法(带奖品送完判断和每用户最大中奖判断) 2 #-------网上的抽奖算法Php 3 #----java版本的.. 4 参考 5 #---抽奖算法需要满足的需求如下: 1.可以控制中奖的概率 2.具有随机性 3.最好可以控制奖品的数量 4.根据用户ID或者ip.手机号.QQ号等条件限制抽奖次数 初期就这些需求,然后根据网上的资料,采用了一种阶段式抽取的方法,大家下面

java版根据权重抽奖算法

根据权重进行抽取的算法应用比较广泛,其中抽奖便是主要用途之一.正好这几天也正在进行抽奖模块的开发,整个抽奖模块涉及到的地方大概有三处,分别是后台进行奖品的添加(同时设置权重和数量),前台根据后台配置生成抽奖队列并根据指令开始抽奖活动,最后一部分是后台统计中奖情况并设置物流状态.本文主要针对前台抽奖算法进行介绍如何根据权重设置每个奖品被抽到的概率. 抽奖算法的核心是根据权重设置随机数出现的概率,在此我将它封装成一个生成随机数的随机类,代码如下: /** * JAVA 返回随机数,并根据概率.比率

抽奖功能实现

1. 需求分析 抽奖可以获得积分,礼券,小样,正品等 若库存为0,则用户不能在抽中此奖 每个奖项的中奖概率 每天抽奖次数的限制 每次抽奖是否需要消耗积分的限制 有没有批量抽奖功能 2.表结构设计 会员表,积分表,是以前就有的,本次新增抽奖功能,需要新增一下的表: 首先要有库存表,暂且定义为BPRIZE表 其次要记录用户抽中的奖项,暂且定义为HPRIZE表        3. 后台代码实现 4. 前端UI实现 原文地址:https://www.cnblogs.com/qianjinyan/p/11

如何用 Python 写一个简易的抽奖程序

不知道有多少人是被这个头图骗进来的:) 事情的起因是这样的,上周有同学问小编,看着小编的示例代码敲代码,感觉自己也会写了,如果不看的话,七七八八可能也写的出来,但是一旦自己独立写一段程序,感觉到无从下手. 其实这个很正常,刚开始学习写代码,都是跟着别人的套路往下写,看的套路少,很难形成自己的套路,这就和做数学题是一样的,做一道题就想会所有的题目,这个可能性微乎其微,都是通过大量的练习来摸索到自己的套路. 正好快过年了,各个公司都会搞一些抽奖活动,小编今天就来聊一下,如果要写一个简单的抽奖程序,小

领域驱动设计(DDD)

领域驱动设计的概念 大家都知道软件开发不是一蹴而就的事情,我们不可能在不了解产品(或行业领域)的前提下进行软件开发,在开发前通常需要进行大量的业务知识梳理,然后才能到软件设计的层面,最后才是开发.而在业务知识梳理的过程中,必然会形成某个领域知识,根据领域知识来一步步驱动软件设计,就是领域驱动设计(DDD,Domain-Driven Design)的基本概念 . 为什么需要 DDD 在业务初期,功能大都非常简单,普通的 CRUD 就基本能满足要求,此时系统是清晰的.但随着产品的不断迭代和演化,业务

javascript抽奖插件+概率计算

写了一个抽奖的jquery插件和计算概率的方法, 结合起来就是一个简单的概率抽奖, 不过实际项目中基本不会把抽奖概率的计算放在前端处理~. 概率计算 function Probability(conf) { this.probArr = conf || []; this.range = [], this.len = this.probArr.length; if (this.len > 0) { this.init(); } } Probability.prototype = { init: f