测验 Web 应用程序错误:生成 10 个随机问题的函数有时会生成 9 或 11 个问题



我正在使用flask在pythonanywhere上编写一个测验应用程序。这是我第一次在任何地方使用烧瓶或python,所以我仍然在学习。下面的函数抛出了一个奇怪的错误,有时它会生成 11 或 9 个字典条目而不是 10 个,即使 Qnum 参数永远不会改变。

我认为问题可能与别名有关(因为该函数会删除条目),因此我尝试通过向下浏览字典键和值来制作单独的列表。当我直接在我的主应用程序文件中编写它时,代码运行良好,但是一旦我将其抽象为帮助程序函数,它就开始播放。

从帮助程序函数文件:

def create_answer_dict(Dict, Qnum):
import random
Qdict={}
for i in range(Qnum):
#Choose random word to test
Qkeys=[]
for key in Dict.keys():
Qkeys.append(key)
Qword=random.choice(Qkeys)
#Get correct answer from dictionary
correctAnswer = Dict[Qword]
#Generate wrong answer options
wrongAnswers=[]
for value in Dict.values():
wrongAnswers.append(value)
del wrongAnswers[wrongAnswers.index(correctAnswer)]
wrongAnswers = random.sample(wrongAnswers, 3)
answerOptions = wrongAnswers + [correctAnswer]
random.shuffle(answerOptions)
Qdict[Qword]=answerOptions
return Qdict

从主应用文件:

@app.route("/", methods=["GET","POST"])
def index():
Qdict=create_answer_dict(questions, total)
if request.method == "GET":
return render_template('main.html', q = Qdict, keys=Qdict.keys())
elif request.method == 'POST':
score=0
for i in Qdict.keys():
answered=request.form[i]
if original_questions[i]==answered:
score+=1
return render_template("results.html", score=score, total=total)

从 html 视图:

<form action='/' method='POST'>
<ol>
{% for i in keys %}
<li>What is the French for <u>{{i}}</u> ?  </li>
{% for j in q[i] %}
<input type='radio' value='{{j}}' name='{{i}}'      style="margin-right: 5"/>{{j}}
<br></br>
{% endfor %}
{% endfor %}
</ol>
<input type="submit" value="submit" />
</form>

它应该如何工作:

可能的问题和答案存储在字典对象中。

在我的主应用程序文件中,我使用我的问答字典和变量总计作为参数,从帮助程序函数文件中调用此函数。总计设置为 10。

该函数选择 Qnum 问题,找到相应的答案,并选择 3 个随机错误答案。

它以以下格式将这些作为字典返回:

{Question1:[CorrectAnswer, IncorrectAnswer1,IncorrectAnswer2, IncorrectAnswer3],
Question2:[CorrectAnswer, IncorrectAnswer1,IncorrectAnswer2, IncorrectAnswer3], 
etc.}

一切都回来了,不会引发错误,只是有时字典比预期少一个或多一个条目。

您不能期望通过从另一个字典中随机选择n条目来获得长度n的字典,因为总是有可能选择重复的条目(并且由于字典键是唯一的,因此生成的字典中将覆盖重复的条目)。

在字典中选取固定数量的随机键n更好的方法是简单地从字典键创建一个列表,随机播放该列表,然后对该列表进行切片以仅保留前n元素。

在您的代码中,它看起来像这样:

def create_answer_dict(Dict, Qnum):
import random
Qdict={}
possibleQuestions = list(Dict.keys())
random.shuffle(possibleQuestions)
possibleQuestions = possibleQuestions[:Qnum]
for Qword in possibleQuestions:
#Get correct answer from dictionary
correctAnswer = Dict[Qword]
#Generate wrong answer options
wrongAnswers = list(Dict.values())
del wrongAnswers[wrongAnswers.index(correctAnswer)]
wrongAnswers = random.sample(wrongAnswers, 3)
answerOptions = wrongAnswers + [correctAnswer]
random.shuffle(answerOptions)
Qdict[Qword] = answerOptions
return Qdict

这将保证生成Qnum独特的问题。

编辑:另外,在index()中,为了避免KeyErrors,如果用户没有回答所有问题,请替换

for i in Qdict.keys():
answered=request.form[i]
...

for i in request.form:
answered=request.form[i]
...

工作演示:https://repl.it/@glhr/55701832

最新更新