原标题:利用Python写一个抽奖程序,解密游戏内抽奖的秘密

前言

本文的文字及图片来源于网络,仅供学习、交流使用,不具有任何商业用途,版权归原作者所有,如有问题请及时联系我们以作处理。

作者: 极客挖掘机

PS:如有需要Python学习资料的小伙伴可以加点击下方链接自行获取

http://note.youdao.com/noteshare?id=3054cce4add8a909e784ad934f956cef

分析需求

我们先整理下思路,目标是什么?

目标是要写一个抽奖程序,那么抽奖程序的核心是什么?

当然是如何判断一个人中奖了。那么如何判断一个人中奖呢?

是不是可以通过随机函数来操作呢?

中奖方法

一步一步来,我们先通过随机函数来判断是否中奖。代码是不是可以先写成下面这样:

import random

# 判断中奖函数

def lottery():

flag = random.randint(0, 9)

if flag < 2:

return True

else:

return False

首先,我们获取 0 ~ 9 之间的随机正整数(这里不讨论 random 是不是真随机,从狭义上来讲我们可以认为它是随机的),如果中奖率为 20% 的话,我们可以认为小于 2 的数字为中奖,其余的为没有中奖。然后中奖后返回 True ,没有中奖返回 False 。

我们加一个入口测试函数,测试一下上面的代码是否能正常运行,并且中奖率是否能维持在大约 20 % 左右。

if __name__ == '__main__':

# 中奖次数

a = 0

# 没有中奖次数

b = 0

for i in range(1000000) :

if (lottery()):

a += 1

else:

b += 1

print('共计中奖:', a, ',未中奖:', b)

执行结果:

共计中奖: 200145 ,未中奖: 799855

上面的测试总共循环了 1 百万次,大约执行需要 2 ~ 3 秒左右,速度还是蛮快的。可以看到,中奖结果确实接近 20% 左右。

动态中奖率

难道到这里就结束了么?当然不可能,这里只是刚刚开了个头。

如果这时老板说,你这个概率不能调整啊,需要让中奖率可以动态调整的,活动刚开始的时候中奖率要高,随着时间的推移,中奖率要降下来。

这时候咋整,傻眼了吧。

既然中奖率要可调整,那么我们中奖率就不能定死在程序中了,这个中奖率需要有一个地方去做存储,在每次做随机的时候将这个中奖率取出来。

简单易行的方法就是将这个中奖率放在数据库中或者缓存服务中,这个根据实际业务场景来定。一般是根据预估访问压力的大小来进行技术选型,如果压力不是特别大,那么放在数据库中也是可以的,如果并发会比较高的话,建议还是放在缓存中。

我们来写一个从数据库获取中奖概率的方法(为了展示直观,小编这里直接使用 Mysql 数据库用作数据存储),先看下数据库的数据:

很简单的设计了一张表,里面有意义的字段有两个,一个用作中奖率的分子部分,一个用作中奖率的分母部分。分母部分最好要设置成 100 、 1000 、 10000 这种,这样计算中奖率会比较好计算。

def get_lottery_rate():

conn = pymysql.connect(host='localhost', user='root', password='password', database='test', charset='utf8mb4')

try:

sql = 'SELECT fenzi, fenmu FROM rate'

cursor = conn.cursor()

cursor.execute(sql)

result = cursor.fetchone()

return result

except Exception as ex:

print(ex)

finally:

conn.close()

运行这个方法测试结果如下:

(10, 100)

可以看到,我们获得了一个元组,里面的内容就是我们从数据库取出来的分子和分母。

我们将前面的抽奖的那个方法改一下,改成从数据库获取中奖比例。修改后的代码如下:

def lottery():

rate = get_lottery_rate()

flag = random.randint(1, rate[1])

if flag < rate[0]:

return True

else:

return False

还是运行上面的测试方法,这里要注意下,因为我们现在是从数据库获取数据,每次方法执行都要加上数据库链接的建立与销毁,建议将循环次数修改为 1000 以内,不然执行的时间就有点太长了。

小编这里将循环次数修改为 1000 次后,执行结果如下:

共计中奖: 92 ,未中奖: 908

那么到这里,我们就可以通过修改数据库中数据实时的操作中奖率了。当然上面的慢的问题我们可以使用数据库连接池等技术进行优化。

增加奖项

那么是否就结束了呢?no no no,我们接着加需求。

现在,我们只能知道每次到底中不中奖,只有一个奖项,但是现在想变成 3 个奖项,如:一等奖、二等奖、三等奖那该怎么办?

这个对之前的抽奖方法改动就有点大了,首先我们先在数据库增加出来另外两个奖项的配置:

配置这里三个奖项的分母最好保持一致,否则后续计算会徒增复杂度。

修改我们获取配置的那个方法:

def get_lottery_rate():

conn = pymysql.connect(host='localhost', port = 3306, user='root', password='password', database='test', charset='utf8mb4')

try:

sql = 'SELECT * FROM rate order by id asc '

cursor = conn.cursor()

cursor.execute(sql)

result = cursor.fetchall()

return result

except Exception as ex:

print(ex)

finally:

conn.close()

测试调用后结果如下:

((1, 10, 100), (2, 5, 100), (3, 1, 100))

先在我们要做的是要将这个配置融入进我们之前的中奖的那个方法中,不多说,直接上代码:

# 判断中奖函数

def lottery():

config = get_lottery_rate()

flag = random.randint(1, config[0][2])

if flag <= config[0][1]:

return 1

elif flag > config[0][1] and flag <= (config[1][1] + config[0][1]):

return 2

elif flag > (config[1][1] + config[0][1]) and flag <= (config[2][1] + config[1][1]):

return 3

else:

return 0

接着修改我们的做测试的代码:

def main():

# 一等奖中奖次数

a = 0

# 二等奖中奖次数

b = 0

# 三等奖中奖次数

c = 0

# 未中奖次数

d = 0

# 循环次数

e = 0

for i in range(1000):

e += 1

print('当前循环次数:', e)

result = lottery()

print('当前中奖结果:', result)

if (result == 1):

a += 1

elif (result == 2):

b += 1

elif (result == 3):

c += 1

else:

d += 1

print('一等奖中奖:', a, ',二等奖中奖次数:', b, ',三等奖中奖次数:', c, ',未中奖次数:', d)

调用我们的测试方法:

if __name__ == '__main__':

main()

小编这里的运行结果如下:

增加会员判断

到这里我们还没完,还能加需求,现在网站大多数都是会员制的,比如白银会员,黄金会员,钻石会员,如果不同的会员等级需要有不同的中奖率,这个是很正常的一件事儿,小编现在还清晰的记得当年某家大型互联网公司代码中的注释 “穷逼 VIP(活动送的那种)” 。

我们假设钻石会员的中奖率为整体中奖率的 100% ,黄金会员的中奖率为整体中奖率的 50% ,白银会员的中奖率为整体中奖率的 20% 。

最简单的实现方式是直接在最外层套一层会员中奖率的判断,不知道各位同学怎么想。

小编这里给出自己的解决方案:

# 判断会员等级中奖率过滤

# 会员等级 1.白银会员 2.黄金会员 3. 钻石会员

def vip_lottery(level):

rate = random.randint(1, 10)

# 如果是钻石会员,直接进入抽奖函数

if level == 3:

return lottery()

# 如果是黄金会员, 50% 概率进入抽奖函数

elif level == 2:

if rate <= 5:

return lottery()

else:

return 0

# 如果是白银会员, 20% 概率进入抽奖函数

elif level == 1:

if rate <= 2:

return lottery()

else:

return 0

# 如果是其他,直接返回未中奖

else:

return 0

我们新增一个测试增加会员过滤的测试方法:

# 会员制中奖测试方法

def test_vip():

print('请输入您当前的会员等级:1.白银会员 2.黄金会员 3. 钻石会员')

level = input()

result = vip_lottery(int(level))

if (result == 1):

print('恭喜您中了一等奖')

elif (result == 2):

print('恭喜您中了二等奖')

elif (result == 3):

print('恭喜您中了三等奖')

else:

print('未中奖,谢谢惠顾')

在我们的入口函数中调用这个方法:

if __name__ == '__main__':

test_vip()

最终测试结果如下:

小编的人品还可以嘛,直接就能中三等奖。

import random

import pymysql

# 获取中奖配置

def get_lottery_rate():

conn = pymysql.connect(host='114.67.111.196', port = 3306, user='root', password='wsy@123456', database='test', charset='utf8mb4')

try:

sql = 'SELECT * FROM rate order by id asc '

cursor = conn.cursor()

cursor.execute(sql)

result = cursor.fetchall()

return result

except Exception as ex:

print(ex)

finally:

conn.close()

# 判断中奖函数

def lottery():

config = get_lottery_rate()

flag = random.randint(1, config[0][2])

if flag <= config[0][1]:

return 1

elif flag > config[0][1] and flag <= (config[1][1] + config[0][1]):

return 2

elif flag > (config[1][1] + config[0][1]) and flag <= (config[2][1] + config[1][1]):

return 3

else:

return 0

# 判断会员等级中奖率过滤

# 会员等级 1.白银会员 2.黄金会员 3. 钻石会员

def vip_lottery(level):

rate = random.randint(1, 10)

# 如果是钻石会员,直接进入抽奖函数

if level == 3:

return lottery()

# 如果是黄金会员, 50% 概率进入抽奖函数

elif level == 2:

if rate <= 5:

return lottery()

else:

return 0

# 如果是白银会员, 20% 概率进入抽奖函数

elif level == 1:

if rate <= 2:

return lottery()

else:

return 0

# 如果是其他,直接返回未中奖

else:

return 0

# 批量测试方法

def test():

# 一等奖中奖次数

a = 0

# 二等奖中奖次数

b = 0

# 三等奖中奖次数

c = 0

# 未中奖次数

d = 0

# 循环次数

e = 0

for i in range(1000):

e += 1

print('当前循环次数:', e)

result = lottery()

print('当前中奖结果:', result)

if (result == 1):

a += 1

elif (result == 2):

b += 1

elif (result == 3):

c += 1

else:

d += 1

print('一等奖中奖:', a, ',二等奖中奖次数:', b, ',三等奖中奖次数:', c, ',未中奖次数:', d)

# 会员制中奖测试方法

def test_vip():

print('请输入您当前的会员等级:1.白银会员 2.黄金会员 3. 钻石会员')

level = input()

result = vip_lottery(int(level))

if (result == 1):

print('恭喜您中了一等奖')

elif (result == 2):

print('恭喜您中了二等奖')

elif (result == 3):

print('恭喜您中了三等奖')

else:

print('未中奖,谢谢惠顾')

if __name__ == '__main__':

test_vip()返回搜狐,查看更多

责任编辑:

python抽奖游戏_利用Python写一个抽奖程序,解密游戏内抽奖的秘密相关推荐

  1. 学了C语言,如何利用CURL写一个下载程序?—用nmake编译CURL并安装

    在这一系列的前一篇文章学了C语言,如何为下载狂人写一个磁盘剩余容量监控程序?中,我们为下载狂人写了一个程序来监视磁盘的剩余容量,防止下载的东西撑爆了硬盘.可是,这两天,他又抱怨他的下载程序不好用,让我 ...

  2. python是一门面向什么的语言用词语填空_使用pygame写一个古诗词填空通关游戏

    之前写的诗词填空的游戏支持python2,现在对程序进行了修改,兼容支持python2和python3,附下效果图. 下面是两个主程序 idiom_lib.py代码: # -*- coding=utf ...

  3. python抽奖教程_利用Python写一个抽奖程序,解密游戏内抽奖的秘密|python基础教程|python入门|python教程...

    https://www.xin3721.com/eschool/pythonxin3721/ 分析需求 我们先整理下思路,目标是什么?目标是要写一个抽奖程序,那么抽奖程序的核心是什么?当然是如何判断一 ...

  4. 使用python加PyQt5,利用QMediaPlayer写一个简易的音乐播放器(进度条拖动,音量改变,播放停止切换,歌曲列表))

    当你学习了python之后,总想着利用它去做些什么,无论是制作小工具还是小游戏,都是一种锻炼. 那么,利用python加上PyQt5写一个简单的音乐播放器,可能会是一个有趣的体验. 下面我会分享一下如 ...

  5. 利用python制作拼图_利用python制作拼图小游戏的全过程

    开发工具 Python版本:3.6.4 相关模块: pygame模块: 以及一些Python自带的模块 关注公众号:Python学习指南,回复"拼图"即可获取源码 环境搭建 安装P ...

  6. python汇率转换_利用Python中的Xpath实现一个在线汇率转换器

    前言 在之前的语法里面,我们记得有一个初识Python之汇率转换篇,在那个程序里面我们发现可以运用一些基础的语法写一个汇率计算,但是学到后面的小伙伴就会发现这个小程序有一定的弊端. 首先,它不可以实时 ...

  7. python画猫和老鼠_利用python如何实现猫捉老鼠小游戏

    python实现猫捉老鼠小游戏 首界面 开始游戏界面 然后键盘操作小老鼠上下左右移动,猫自己去追,当猫追上老鼠则游戏结束 这里用时3.2秒,最后将游戏时长与猫和老鼠都显示在主页面上 下面我把猫与老鼠的 ...

  8. 用python画猫和老鼠_利用python如何实现猫捉老鼠小游戏

    python实现猫捉老鼠小游戏 首界面 开始游戏界面 然后键盘操作小老鼠上下左右移动,猫自己去追,当猫追上老鼠则游戏结束 这里用时3.2秒,最后将游戏时长与猫和老鼠都显示在主页面上 下面我把猫与老鼠的 ...

  9. python自制电子记事本_利用Python制作一个“电子记事本”

    案例内容 今天的挑战就是写一个"记事本"小程序.程序的功能分为三个部分: 1.把内容记录到文件. 2.显示记录的所有内容. 3.删除不再需要的内容. 正式的"记事本&qu ...

最新文章

  1. 2010.4.18 OA 项目组一周工作报告
  2. GetSafeHwnd()函数
  3. 和同学沟通,一定是时间效率比较高的
  4. ng-repeat循环出来的部分调用同一个函数并且实现每个模块之间不能相互干扰
  5. Android嵌入式安卓触摸屏|4418开发板平台
  6. 20200227:存在重复元素Ⅱ(leetcode219)
  7. 给大家推荐一款非常好用的表单验证插件:lr-verify.js
  8. Docker下Cannot connect to the Docker daemon. Is the docker daemon running on this host错误解决方案
  9. unistd.h中定义函数
  10. CodeForces 621C Wet Shark and Flowers
  11. ufs qfil注意事项
  12. 远控免杀专题10--TheFatRat免杀
  13. 接口安全评估基本流程
  14. 分析武汉二手房房价信息
  15. Android刘海屏、水滴屏全面屏适配详解,997页字节跳动Android面试真题解析火爆全网
  16. Python基于PyTorch实现BP神经网络ANN分类模型项目实战
  17. 数据结构 期末复习主观题练习题(答案版)
  18. python逗号表达式_正则表达式 - 在第一个逗号前获取所有内容。 - python
  19. C语言 % x的作用,关于c语言%#X意思大全
  20. 1.1.5 计算机网络的标准化工作及相关组织

热门文章

  1. Total Control 远程控制手机软件 和 Vysor Chrome 的插件实时同步投影
  2. 《Mall商城的设计与实现》软件工程综合实践 课程设计
  3. 手术麻醉临床信息管理系统源码 医院手麻系统源码 B/S架构 大屏幕 三甲医院
  4. 提示 api-ms-win-crt-runtime-l1-1-0.dll丢失 ivms-4200 安装好运行报错api-ms-win-crt-runtime-|1-1-0.dll
  5. 计算机控制系统的综合化,[工学]计算机控制系统.ppt
  6. React 系列教程
  7. 关于在Ubuntu18.04 kernel4.8中安装rtl8192fu驱动的一些问题
  8. dubbo控制台tomcat启动出错
  9. JAVA 线程池工作原理 图解
  10. Excel VBA 实例