码迷,mamicode.com
首页 > 其他好文 > 详细

预测球队比赛成绩

时间:2020-04-19 14:47:56      阅读:55      评论:0      收藏:0      [点我收藏+]

标签:运行   团队   inpu   理想   连续   main   比较   turn   思路   

分析乒乓球比赛所设计出的代码:

 
from random import random
def printIntro():
    print("这个程序模拟两个选手A和B的某种比赛")
    print("程序运行需要A和B的能力值(0到1之间)")
def getInputs():
    a=eval(input("请输入选手A的能力值(0-1):"))
    b=eval(input("请输入选手B的能力值(0-1):"))
    n=eval(input("模拟比赛的场次:"))
    return a,b,n
def simNGames(n,probA,probB):
    winsA,winsB=0,0
    for i in range(n):
        scoreA,scoreB=simOneGame(probA,probB)
        if scoreA>scoreB:
            winsA+=1
        else:
            winsB+=1
    return winsA,winsB
def gameOver(a,b):
    return a==11 or b==11
def simOneGame(probA,probB):
    scoreA,scoreB = 0,0
    serving = ‘A‘
    while not gameOver(scoreA,scoreB):
        if serving == ‘A‘:
            if random()<probA:
                scoreA+=1
            else:
                serving=‘B‘
        else:
            if random()<probB:
                scoreB+=1
            else:
                serving=‘A‘
    return scoreA,scoreB
def printSummary(winsA,winsB):
    n=winsA+winsB
    print("竞技分析开始,共模拟{}场比赛".format(n))
    print("选手A获胜{}场比赛,占比{:.1%}".format(winsA,winsA/n))
    print("选手B获胜{}场比赛,占比{:.1%}".format(winsB,winsB/n))
def main():
    printIntro()
    probA,probB,n = getInputs()
    winsA,winsB = simNGames(n,probA,probB)
    printSummary(winsA,winsB)
main()
 

 根据以上的代码我们整理一下思路,这是一段通过“顶层设计”编写出来的代码,通过一些封装,我们对不同的动作进行解释

我们了解到一局乒乓球比赛的规则是,发球方发球,进行“两球制”发球,即两人固定轮流连续开两球,不论胜负,谁分数先到达11分即胜利

由于还有考虑最后平手时加球的情况,但上述代码暂不考虑,所以我们来看看代码的运算效果:
技术图片

输入两个选手的能力值,我设置为各0.5,但结果虽较接近但还是有些许差距,排除random()的误差,我们知道这是比较符合现实的一种情况!

我们再试试能力值有细微差距的情况:

 技术图片

wow!看来仅仅只是一点点的实力差距就会有天壤之别的比赛结果……

我们再深入分析,加入更多的因素,我们知道乒乓球的赛制时单打七局四胜制,双打则是五局三胜制,所以我们先用以下代码预测单打情况:

 
from random import random
def printIntro():
    print("这个程序模拟两个选手A和B的某种比赛")
    print("程序运行需要A和B的能力值(0到1之间)")
def getInputs():
    a=eval(input("请输入选手A的能力值(0-1):"))
    b=eval(input("请输入选手B的能力值(0-1):"))
    return a,b
def simNGames(probA,probB):
    winsA,winsB=0,0
    for i in range(7):
        scoreA,scoreB=simOneGame(probA,probB)
        if scoreA>scoreB:
            winsA+=1
            if winsA==4:
                break
        else:
            winsB+=1
            if winsB==4:
                break
    return winsA,winsB
def gameOver(a,b):
    return a==11 or b==11
def simOneGame(probA,probB):
    scoreA,scoreB = 0,0
    serving = ‘A‘
    while not gameOver(scoreA,scoreB):
        if serving == ‘A‘:
            if random()<probA:
                scoreA+=1
            else:
                serving=‘B‘
        else:
            if random()<probB:
                scoreB+=1
            else:
                serving=‘A‘
    return scoreA,scoreB
def printSummary(winsA,winsB):
    n=winsA+winsB
    print("竞技分析开始,共模拟{}场比赛".format(n))
    print("选手A获胜{}场比赛,占比{:.1%}".format(winsA,winsA/n))
    print("选手B获胜{}场比赛,占比{:.1%}".format(winsB,winsB/n))
def main():
    printIntro()
    probA,probB = getInputs()
    winsA,winsB = simNGames(probA,probB)
    printSummary(winsA,winsB)
main()
 

我们再试试结果:

技术图片

我们看到,A选手最终以4-2击败B选手,即使他们能力几乎相当,也许他当下更有手感吧~

接下来是双打时,我们将刚刚的A/B选手抽象为A/B队伍,能力值为队伍的平均能力值。

 
from random import random
def printIntro():
    print("这个程序模拟两个选手A和B的某种比赛")
    print("程序运行需要A和B的能力值(0到1之间)")
def getInputs():
    a=eval(input("请输入选手A的能力值(0-1):"))
    b=eval(input("请输入选手B的能力值(0-1):"))
    return a,b
def simNGames(probA,probB):
    winsA,winsB=0,0
    for i in range(5):
        scoreA,scoreB=simOneGame(probA,probB)
        if scoreA>scoreB:
            winsA+=1
            if winsA==3:
                break
        else:
            winsB+=1
            if winsB==3:
                break
    return winsA,winsB
def gameOver(a,b):
    return a==11 or b==11
def simOneGame(probA,probB):
    scoreA,scoreB = 0,0
    serving = ‘A‘
    while not gameOver(scoreA,scoreB):
        if serving == ‘A‘:
            if random()<probA:
                scoreA+=1
            else:
                serving=‘B‘
        else:
            if random()<probB:
                scoreB+=1
            else:
                serving=‘A‘
    return scoreA,scoreB
def printSummary(winsA,winsB):
    n=winsA+winsB
    print("竞技分析开始,共模拟{}场比赛".format(n))
    print("选手A获胜{}场比赛,占比{:.1%}".format(winsA,winsA/n))
    print("选手B获胜{}场比赛,占比{:.1%}".format(winsB,winsB/n))
def main():
    printIntro()
    probA,probB = getInputs()
    winsA,winsB = simNGames(probA,probB)
    printSummary(winsA,winsB)
main()
 

结果:A队伍以3-2赢得双打比赛~

技术图片

接下来我们分析篮球赛事,我们知道篮球是一项团队的运动,所以左右比赛的走向的的变量更加多首先是运动员的人数较多,场上每一队应该有5名队员在场,场下有4-5名替补队员,这样加起来双方队伍参赛队员将在14-20个,同时能力值有不同,且若精确到每一次进攻,那么就会有进攻成功及进攻失败,或进攻一次伴随着抢板之后再次补篮等多种复杂情况……由于现实的多变性,考虑到本次只是简单的模拟,我对现实进行理想化,只讨论场上每队主力5名队员的能力值,并通过各值推出该队平均能力值,由于篮球比赛以时间作为指标,在指定时间内获得分数最高者胜,所以我们将其简化为:在一定时间内,只能有有限次的进攻次数,根据现实情况,我们设置为:200次(双方进攻总数),于是有了下面的代码:

 
from random import random
import time as T

def printIntro():
    print("这个程序模拟两支球队A和B的篮球比赛")
    print("程序运行需要A队和B队的能力值(0到1之间)")
def getInputs():
    print("现在请分别输入A队中上场的五位球员的能力值")
    a=eval(input("请输入选手1的能力值(0-1):"))
    b=eval(input("请输入选手2的能力值(0-1):"))
    c=eval(input("请输入选手3的能力值(0-1):"))
    d=eval(input("请输入选手4的能力值(0-1):"))
    e=eval(input("请输入选手5的能力值(0-1):"))
    print("现在请分别输入B队中上场的五位球员的能力值")
    f=eval(input("请输入选手6的能力值(0-1):"))
    g=eval(input("请输入选手7的能力值(0-1):"))
    h=eval(input("请输入选手8的能力值(0-1):"))
    i=eval(input("请输入选手9的能力值(0-1):"))
    j=eval(input("请输入选手10的能力值(0-1):"))
    meanA=(a+b+c+d+e)/5
    meanB=(f+g+h+i+j)/5
    print("A队的平均能力值为{0},B队的平均能力值为{1}".format(meanA,meanB))
    return meanA,meanB
def simOneAttack(probA,probB,):
    scoreA,scoreB = 0,0
    serving = ‘A‘
    for i in range(200):
        if serving == ‘A‘:
            if random()<probA:
                scoreA+=2
            else:
                serving=‘B‘
        else:
            if random()<probB:
                scoreB+=2
            else:
                serving=‘A‘
    print("最终A队得分为{0},B队为{1}".format(scoreA,scoreB))
    print("队伍A获胜概率为{:.1%}".format(scoreA/200))
    print("队伍B获胜概率为{:.1%}".format(scoreB/200))
    print("平手概率(不加时):{:.1%}".format(1-(scoreA/200)-(scoreB/200)))
    return scoreA,scoreB 
def main():
    printIntro()
    probA,probB = getInputs()
    print("竞技分析开始,共模拟200次进攻")

    simOneAttack(probA,probB)
main()
 

结果如下图:

技术图片

分别输入各个队伍主力球员的能力值,求出各队伍平均能力值,并最终预测比赛得分为:A:B=86:110

这是我们预测了一次比赛的结果。

我们继续分析,在NBA的规则中,需要先在本土30支队伍中通过进行82场常规赛决出8强,于是我们对其进行变体来预测A/B两支队伍进行多场比赛后的结果:

 
from random import random
import time as T

def printIntro():
    print("这个程序模拟两支球队A和B的篮球比赛")
    print("程序运行需要A队和B队的能力值(0到1之间)")
def getInputs():
    print("现在请分别输入A队中上场的五位球员的能力值")
    a=eval(input("请输入选手1的能力值(0-1):"))
    b=eval(input("请输入选手2的能力值(0-1):"))
    c=eval(input("请输入选手3的能力值(0-1):"))
    d=eval(input("请输入选手4的能力值(0-1):"))
    e=eval(input("请输入选手5的能力值(0-1):"))
    print("现在请分别输入B队中上场的五位球员的能力值")
    f=eval(input("请输入选手6的能力值(0-1):"))
    g=eval(input("请输入选手7的能力值(0-1):"))
    h=eval(input("请输入选手8的能力值(0-1):"))
    i=eval(input("请输入选手9的能力值(0-1):"))
    j=eval(input("请输入选手10的能力值(0-1):"))
    meanA=(a+b+c+d+e)/5
    meanB=(f+g+h+i+j)/5
    print("A队的平均能力值为{0},B队的平均能力值为{1}".format(meanA,meanB))
    return meanA,meanB
def simNGames(probA,probB):
    winsA,winsB=0,0
    for i in range(82):
        scoreA,scoreB=simOneAttack(probA,probB)
        if scoreA>scoreB:
            winsA+=1
        else:
            winsB+=1
    print("最终A队得分为{0},B队为{1}".format(scoreA,scoreB))
    return winsA,winsB
def simOneAttack(probA,probB,):
    scoreA,scoreB = 0,0
    serving = ‘A‘
    for i in range(200):
        if serving == ‘A‘:
            if random()<probA:
                scoreA+=2
            else:
                serving=‘B‘
        else:
            if random()<probB:
                scoreB+=2
            else:
                serving=‘A‘

    return scoreA,scoreB
def printSummary(winsA,winsB):
    n=winsA+winsB
    print("竞技分析开始,共模拟{}场比赛".format(82))
    print("队伍A获胜{}场比赛,占比{:.1%}".format(winsA,winsA/n))
    print("队伍B获胜{}场比赛,占比{:.1%}".format(winsB,winsB/n))
def main():
    printIntro()
    probA,probB = getInputs()
    print("竞技分析开始,共模拟200次进攻")
    simOneAttack(probA,probB)
    winsA,winsB = simNGames(probA,probB)
    printSummary(winsA,winsB)
main()
 

结果:

技术图片

 

预测球队比赛成绩

标签:运行   团队   inpu   理想   连续   main   比较   turn   思路   

原文地址:https://www.cnblogs.com/567823a/p/12731215.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!