分享
  1. 首页
  2. 文章

抽奖问题分析

tomorrowwu · · 3030 次点击 · · 开始浏览
这是一个创建于 的文章,其中的信息可能已经有所发展或是发生改变。

普通抽奖问题

问题描述

用户随机抽奖,数据如下:

// map中,key代表用户名,value代表成用户下单数
var users map[string]int64 = map[string]int64{
 "a": 10,
 "b": 6,
 "c": 3,
 "d": 12,
 "f": 1,
}

思路

随机问题,一般就是通过随机函数从某个范围内随机取出某个数值,则该数值对应的就是中奖用户

在这里,如果我们能给map中每个元素设置对应的索引,即转化为数组,是不是就可以解决问题了呢?

代码实现

func GetAwardUserName(users map[string]int64) (name string) {
 size := len(users)
 awardIndex := rand.Intn(size)
 i := 0
 for userName, _ := range users {
 if i == awardIndex {
 name = userName
 return
 }
 i++
 }
 return
}

单元测试

func Test_GetAwardUserName(t *testing.T) {
 var users map[string]int64 = map[string]int64{
 "a": 10,
 "b": 6,
 "c": 3,
 "d": 12,
 "f": 1,
 }
 rand.Seed(time.Now().Unix())
 awardCount := make(map[string]int)
 for i := 0; i <= 1000000; i++ {
 awardName := GetAwardUserName(users)
 if count, ok := awardCount[awardName]; ok {
 awardCount[awardName] = count + 1
 } else {
 awardCount[awardName] = 0
 }
 }
 for n, c := range awardCount {
 fmt.Printf("%v:%v\n",n,c)
 }
}

测试结果:

为了验证获奖概率的正确性,循环执行100万次,每个用户获奖的次数基本在20万左右,每个用户的获奖概率相等

c:200102
f:199853
b:198942
a:200395
d:200704

权重抽奖

问题描述:

数据结构和上面抽奖问题一致,只是这里,要求中奖概率和用户的订单数成正比

思路

==本质==还是随机函数获得一个数值,数值对应的用户即获奖用户;这里要实现订单数对获奖概率的影响问题,即==订单数对应随机数的某个范围,订单数越大,范围越大,随机数落在范围内的概率越大==

代码实现

func getAwardUser_weight(users map[string]int64) (name string) {
 type awardUser struct {
 name string
 offset int64
 count int64
 }
 userSli := make([]*awardUser, 0,len(users))
 var sumCount int64 = 0
 for n, c := range users {
 a := awardUser{
 name: n,
 offset: sumCount,
 count: c,
 }
 //整理所有用户的count数据为数轴
 userSli = append(userSli, &a)
 sumCount += c
 }
 awardIndex := rand.Int63n(sumCount)
 for _, u := range userSli {
 //判断获奖index落在那个用户区间内
 if u.offset+u.count>awardIndex {
 name = u.name
 return
 }
 }
 return
}

单元测试

func Test_getAwardUser_weight(t *testing.T) {
 var users map[string]int64 = map[string]int64{
 "a": 10,
 "b": 6,
 "c": 3,
 "d": 12,
 "f": 1,
 }
 rand.Seed(time.Now().Unix())
 awardCount := make(map[string]int)
 for i := 0; i <= 100000; i++ {
 awardName := getAwardUser_weight(users)
 if count, ok := awardCount[awardName]; ok {
 awardCount[awardName] = count + 1
 } else {
 awardCount[awardName] = 0
 }
 }
 for n,c := range awardCount {
 fmt.Printf("%v:%v \n",n,c)
 }
}

测试结果:

循环遍历了100万次,获奖的次数,与用户的订单数成正比

c:93479 
f:31206 
d:375614 
b:186933 
a:312764 

总结

解决实际问题,往往都有数学模型去对应,比如抽奖问题,就可以转化为初中所学习的数轴知识,画个草图,简单易理解,也不需要多高深的数学知识

问题本身并不难,重要的是转换思路,将抽象问题简化为具体的数学问题,然后去解决


有疑问加站长微信联系(非本文作者)

本文来自:Segmentfault

感谢作者:tomorrowwu

查看原文:抽奖问题分析

入群交流(和以上内容无关):加入Go大咖交流群,或添加微信:liuxiaoyan-s 备注:入群;或加QQ群:692541889

关注微信
3030 次点击
暂无回复
添加一条新回复 (您需要 后才能回复 没有账号 ?)
  • 请尽量让自己的回复能够对别人有帮助
  • 支持 Markdown 格式, **粗体**、~~删除线~~、`单行代码`
  • 支持 @ 本站用户;支持表情(输入 : 提示),见 Emoji cheat sheet
  • 图片支持拖拽、截图粘贴等方式上传

用户登录

没有账号?注册
(追記) (追記ここまで)

今日阅读排行

    加载中
(追記) (追記ここまで)

一周阅读排行

    加载中

关注我

  • 扫码关注领全套学习资料 关注微信公众号
  • 加入 QQ 群:
    • 192706294(已满)
    • 731990104(已满)
    • 798786647(已满)
    • 729884609(已满)
    • 977810755(已满)
    • 815126783(已满)
    • 812540095(已满)
    • 1006366459(已满)
    • 692541889

  • 关注微信公众号
  • 加入微信群:liuxiaoyan-s,备注入群
  • 也欢迎加入知识星球 Go粉丝们(免费)

给该专栏投稿 写篇新文章

每篇文章有总共有 5 次投稿机会

收入到我管理的专栏 新建专栏