Python选择题训练工具:高效学习、答题回顾与音频朗读一站式体验

一、引言

随着人工智能技术的不断进步,传统的教学方式已经逐渐向智能化、互动化转变。在众多英语测试题型中,选择题作为一种高效的方式被广泛应用于各类培训与考试中。为了帮助学生高效学习与自测,本篇文章将采用Python编写一款基于 Python 开发的选择题训练工具。该工具不仅支持加载 Excel 文件中的题库,还具备题干和选项展示、答题记录、音频朗读等多种功能,旨在为用户提供便捷、高效的学习体验。

二、开发背景

在日常教学过程中,老师往往需要准备大量的选择题作为测试题目。我们可以把这些题目存储在 Excel 文件中,如果手动处理和展示题目会非常繁琐。为了提高教学效率,选用一种自动化展示工具成为一种迫切的需求。本项目通过 Python 和 Tkinter 库实现了一个图形化界面应用,可以自动加载并展示选择题,记录用户答题情况,同时通过音频朗读题目,进一步增强互动性。软件的示意图如下:

软件展示

三、软件特点

  1. 题库加载与展示:支持从 Excel 文件中加载选择题,并且可以根据课程单元名称筛选题目。
  2. 音频朗读功能:使用 Python 的 Pygame 库播放题目音频,即朗读题干,帮助用户通过听力巩固已学单词。
  3. 答题记录与统计:软件会自动记录用户的答题情况,并在决定答题结束时,提供正确率和错误题目的回顾功能。
  4. 多线程支持:使用线程处理音频播放等耗时操作,避免界面卡顿,提升用户体验。同时,当完成一道题时,会自动转到下一道,并给出正确或错误的提示。正确就是显示绿色button选项,错误就显示红色button选项。
  5. 图形化界面:基于 Tkinter 实现的 GUI 界面,简单直观,易于操作。

四、使用方法

1. 安装依赖

在运行程序之前,需要安装一些 Python 库,依赖包括 openpyxl, pygame 和 requests。可以使用以下命令安装所需的库:

pip install openpyxl pygame requests 

2. 准备题库

本程序需要一个 Excel 文件,并命名为【选择题.xlsx】作为题库,Excel 文件应包括如下列:

  • 单元:课程名称或章节
  • 序号:题目的编号
  • 题干:选择题的内容
  • 选项A、B、C、D:四个选择项
  • 正确答案:正确的答案选项

Excel表的内容如下:

Excel表内容

3. 运行程序

运行 Python 程序,tkinter GUI界面会自动加载,用户可以在顶部下拉框选择不同的课程,然后开始答题,结束前查看战绩,回顾错误题目,还可以点朗读通过音频功能听题干。

4. 功能概览

  • 选择题显示:在每道题目显示时,用户可以选择答案。程序会自动判断正确与否,并高亮显示选择的答案。
  • 战绩统计:用户答题完成后,可以查看自己的正确率统计。
  • 错误回顾:用户可以查看答错的题目,并且看到正确答案与自己的选择。
  • 朗读功能:用户可以点击按钮让程序朗读题干。

五、源码展示

以下是该程序的主要代码:

import tkinter as tk
from tkinter import ttk, messagebox
from openpyxl import load_workbook
import random
import os,re
import pygame,requests
from io import BytesIO
from threading import Lock, Thread, Timerdef load_excel_data(filename):workbook = load_workbook(filename)sheet = workbook.activequestions = []lessons = set()  # 用于存储所有课程名,避免重复for row in sheet.iter_rows(min_row=2, values_only=True):lesson = row[0]lessons.add(lesson)question = {'单元': lesson,'序号': row[1],'题干': row[2],'选项': [row[3], row[4], row[5], row[6]],'正确答案': row[7]}questions.append(question)return list(lessons), questionslessons, questions = load_excel_data('选择题.xlsx')
random.shuffle(questions)# 使用正则表达式从课程名中提取数字,并按数字排序课程名
sorted_lessons = sorted(lessons, key=lambda x: (lambda m: int(m.group(0)) if m else 0)(re.search(r'\d+', x)))class QuizApp:def __init__(self, root):self.root = root# 设置窗口大小self.root.geometry("600x500")#self.root.resizable(False, False)pygame.init()self.all_questions = questionsself.questions = questions  # 当前显示的问题列表self.current_question_index = 0self.correct_answers = 0self.wrong_answers = 0self.error_questions = []self.setup_ui()self.load_question()def thread_it(self,func):self.thread1=Thread(target=func)self.thread1.setDaemon(True) self.thread1.start()def setup_ui(self):self.root.title("选择题训练工具")self.lesson_combobox = ttk.Combobox(self.root, values=['全部'] + sorted_lessons, state="readonly")self.lesson_combobox.pack(pady=(10, 0))self.lesson_combobox.set('全部')self.lesson_combobox.bind('<<ComboboxSelected>>', self.on_combobox_change)self.question_label = tk.Label(self.root, text="", font=("Times New Roman", 20), fg="blue", wraplength=520, justify=tk.LEFT) #设置文本长度,并且文本左对齐self.question_label.pack(pady=(10, 40))self.options_frame = tk.Frame(self.root)self.options_frame.pack(pady=20)option_labels = ['A', 'B', 'C', 'D']  # 选项标签self.option_buttons = []  # 存储选项按钮self.option_labels = []  # 存储选项标签控件for i in range(4):# 创建选项标签控件并放置label = tk.Label(self.options_frame, text=f"{option_labels[i]}.", font=("Times New Roman", 16))label.grid(row=i, column=0, pady=5, sticky="e")self.option_labels.append(label)# 创建选项按钮并放置btn = tk.Button(self.options_frame, text="", font=("Times New Roman", 16), width=20,command=lambda b=i: self.check_answer(b),  # type: ignoreanchor="w")btn.grid(row=i, column=1, pady=5)self.option_buttons.append(btn)self.action_frame = tk.Frame(self.root)self.action_frame.pack(side=tk.BOTTOM, pady=20)tk.Button(self.action_frame, text="退出程序", font=("宋体", 16, "bold"), width=9, command=self.ui_quit).pack(side=tk.LEFT, padx=10)tk.Button(self.action_frame, text="我的战绩", font=("宋体", 16, "bold"), width=9, command=self.show_score).pack(side=tk.LEFT, padx=10)tk.Button(self.action_frame, text="查看错误", font=("宋体", 16, "bold"), width=9, command=self.show_errors).pack(side=tk.LEFT, padx=10)tk.Button(self.action_frame, text="朗读题干", font=("宋体", 16, "bold"), width=9, command=self.show_sound).pack(side=tk.LEFT, padx=10)# 其余UI代码与原来相同...def on_combobox_change(self, event):selected_lesson = self.lesson_combobox.get()if selected_lesson == '全部':self.questions = self.all_questionselse:self.questions = [q for q in self.all_questions if q['单元'] == selected_lesson]self.current_question_index = 0self.correct_answers = 0self.wrong_answers = 0self.error_questions = []self.load_question()def load_question(self):if self.current_question_index < len(self.questions):question = self.questions[self.current_question_index]self.question_label.config(text=str(self.current_question_index+1)+". "+question['题干'],anchor='w')correct_answer = question['正确答案']options = question['选项']self.correct_option_index = options.index(correct_answer)random.shuffle(options)for i, option in enumerate(options):self.option_buttons[i].config(text=option, bg='SystemButtonFace')def check_answer(self, button_index):question = self.questions[self.current_question_index]selected_option = self.option_buttons[button_index].cget('text')correct_option = question['正确答案']# 确定正确答案按钮的索引correct_option_index = Nonefor i, btn in enumerate(self.option_buttons):if btn.cget('text') == correct_option:correct_option_index = ibreakif selected_option == correct_option:# 用户选择了正确的答案self.option_buttons[button_index].config(bg='light green')self.correct_answers += 1else:# 用户选择了错误的答案self.option_buttons[button_index].config(bg='red')if correct_option_index is not None:self.option_buttons[correct_option_index].config(bg='light green')self.wrong_answers += 1self.error_questions.append((question['题干'], selected_option, question['正确答案']))# 准备间隔1秒显示下一个问题self.root.after(1000, self.next_question)def next_question(self):self.current_question_index += 1if self.current_question_index < len(self.questions):self.load_question()else:messagebox.showinfo("结束", "所有问题都已回答完毕!")def ui_quit(self):self.root.destroy()def show_score(self):messagebox.showinfo("战绩", f"正确: {self.correct_answers}\n错误: {self.wrong_answers}\n总计: {len(self.questions)}\n正确率:{self.correct_answers/len(self.questions)*100}%")def show_sound(self):self.thread_it(self.show_sound2)def show_errors(self):error_messages = "\n".join([f"题干: {q[0]},正确答案是:{q[2]}, 您的选择: {q[1]}" for q in self.error_questions])messagebox.showinfo("错误回顾", error_messages if error_messages else "完美!没有任何错误。")def show_sound2(self):self.current_name = self.question_label.cget("text")audio_path = f"https://dict.youdao.com/dictvoice?audio={self.current_name}&type=1"resp = requests.get(audio_path)audio_data = BytesIO(resp.content)pygame.mixer.music.load(audio_data)pygame.mixer.music.play()while pygame.mixer.music.get_busy():continue# 退出pygameif __name__ == "__main__":root = tk.Tk()app = QuizApp(root)root.mainloop()

六、注意事项

  1. Excel 文件格式:确保题库文件【选择题.xlsx】的格式正确,每个问题必须包含单元、序号、题干、选项和正确答案等字段。
  2. 音频播放:程序中使用了 Pygame 库进行音频播放,因此需要安装相应的音频库,并且程序会访问在线字典接口生成题目音频。
  3. 线程管理:为了避免界面卡顿,音频播放等操作被放置在单独的线程中处理,保证主界面的流畅性。
  4. 数据存储:答题结果和错误回顾会暂时保存在程序内存中,但如果需要长期保存数据,可以考虑添加导出功能。

七、总结

这款选择题训练工具通过 Python 实现了一个简单易用的答题系统,利用 Tkinter 提供了良好的用户界面,结合 Pygame 和音频播放技术,增强了选择题问答的互动性。它可以帮助学生提高学习效率,帮助教师管理题库,并提供了直观的成绩统计与错误回顾功能,是一款非常实用的教学辅助工具。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/495133.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

《三角洲行动》游戏运行时提示“缺失kernel32.dll”:问题解析与解决方案

《三角洲行动》游戏运行时提示“缺失kernel32.dll”&#xff1a;问题解析与解决方案 作为软件开发领域的一名从业者&#xff0c;我深知电脑游戏运行过程中可能遇到的各种挑战&#xff0c;尤其是文件丢失、文件损坏以及系统报错等问题。今天&#xff0c;我将以经典游戏《三角洲…

【从零开始入门unity游戏开发之——unity篇02】unity6基础入门——软件下载安装、Unity Hub配置、安装unity编辑器、许可证管理

文章目录 一、软件下载安装1、Unity官网2、下载Unity Hub 二、修改Unity Hub配置1、设置Unity Hub中文语言2、修改默认存储目录 三、安装unity编辑器1、点击安装编辑器2、版本选择3、关于版本号4、安装模块选择5、等待下载完成自动安装即可6、追加unity和模块 四、许可证管理专…

AtCoder Beginner Contest 385(A~F)题解

A - Equally 思路&#xff1a;由题可知最多只能分成三组&#xff0c;我们只需要判断是否三个数都相等&#xff0c;或者两个数相加等于另外一个数即可 #include<bits/stdc.h> using namespace std; #define int long long int n; string s; int a,b,c; signed main() {ci…

STM32串口第一次接收数据时第一个字节丢失的问题

解决方法&#xff1a;开启中断之前&#xff0c;先清除标志位【1】。 串口清除标志位&#xff1a; __HAL_UART_CLEAR_PEFLAG(&huart1); HAL_UART_Receive_IT(&huart1,&RxUart, 1); 定时器清除标志位&#xff1a; __HAL_TIM_CLEAR_FLAG(&htim3,TIM_FLAG_UPDATE);…

Unity3d 基于UGUI和VideoPlayer 实现一个多功能视频播放器功能(含源码)

前言 随着Unity3d引擎在数字沙盘、智慧工厂、数字孪生等场景的广泛应用&#xff0c;视频已成为系统程序中展示时&#xff0c;不可或缺的一部分。在 Unity3d 中&#xff0c;我们可以通过强大的 VideoPlayer 组件和灵活的 UGUI 系统&#xff0c;将视频播放功能无缝集成到用户界面…

第22天:信息收集-Web应用各语言框架安全组件联动系统数据特征人工分析识别项目

#知识点 1、信息收集-Web应用-开发框架-识别安全 2、信息收集-Web应用-安全组件-特征分析 一、ICO图标&#xff1a; 1、某个应用系统的标示&#xff0c;如若依系统有自己特点的图标&#xff1b;一旦该系统出问题&#xff0c;使用该系统的网站都会受到影响&#xff1b; 2、某个公…

我的 2024 年终总结

2024 年&#xff0c;我离开了待了两年的互联网公司&#xff0c;来到了一家聚焦教育机器人和激光切割机的公司&#xff0c;没错&#xff0c;是一家硬件公司&#xff0c;从未接触过的领域&#xff0c;但这还不是我今年最重要的里程碑事件 5 月份的时候&#xff0c;正式提出了离职…

acme ssl证书自动续签 nginx

参考 github 官方操作 &#xff0c;acme操作说明 说下我的操作 安装 acme.sh curl https://get.acme.sh | sh source ~/.bashrc 2.注册 acme.sh --register-account -m 123qq.com 如果你在配置 acme.sh 时选择了其他 CA&#xff08;如 Let’s Encrypt&#xff09;&#xff…

sentinel学习笔记6-限流降级(上)

本文属于sentinel学习笔记系列。网上看到吴就业老师的专栏&#xff0c;写的好值得推荐&#xff0c;我整理的有所删减&#xff0c;推荐看原文。 https://blog.csdn.net/baidu_28523317/category_10400605.html sentinel 实现限流降级、熔断降级、黑白名单限流降级、系统自适应…

简单了解函数递归

函数递归 一 了解函数递归二 深入理解函数递归的思想三 函数递归的优缺点 一 了解函数递归 首先&#xff0c;我们通过一个简单的代码来理解函数递归。 #include<stdio.h> int Func() {return Func(n1); } int main() {int n 5;Func(n);return 0; }这个就是函数递归&am…

QT的前景与互联网岗位发展

qt是用来干什么的 --》桌面应用开发&#xff08;做电脑的应用程序&#xff0c;面对客户端&#xff09;。 主要用于开发跨平台的应用程序和用户界面&#xff08;UI&#xff09;。它是一个全面的C库集合&#xff0c;提供了构建软件应用所需的各种工具和功能。 客户端开发的重…

重温设计模式--2、设计模式七大原则

文章目录 1、开闭原则&#xff08;Open - Closed Principle&#xff0c;OCP&#xff09;定义&#xff1a;示例&#xff1a;好处&#xff1a; 2、里氏替换原则&#xff08;Liskov Substitution Principle&#xff0c;LSP&#xff09;定义&#xff1a;示例&#xff1a;好处&#…

第十五章 C++ 数组

C 支持数组数据结构&#xff0c;它可以存储一个固定大小的相同类型元素的顺序集合。数组是用来存储一系列数据&#xff0c;但它往往被认为是一系列相同类型的变量。 数组的声明并不是声明一个个单独的变量&#xff0c;比如 number0、number1、...、number99&#xff0c;而是声…

企业数字化转型加速,现代 IT 如何用 Datadog 全面提升可观测性?

作为 Gartner 可观测平台魔力象限的领导者&#xff0c;Datadog 凭借全面的功能、直观的用户界面和强大的产品路线图赢得了全球企业的信任。 企业 IT 架构正变得日益复杂&#xff0c;从本地服务器到云端部署&#xff0c;从单体应用向微服务&#xff0c;还有容器、 Kubernetes 等…

渗透Vulnhub-DC-9靶机

本篇文章旨在为网络安全渗透测试行业靶机教学。通过阅读本文&#xff0c;读者将能够对渗透Vulnhub系列DC-6靶机有定的了解 一、信息收集阶段 DC-9靶场信息: DC-9靶场介绍&#xff1a; https://www.vulnhub.com/entry/dc-9,412/ DC-9靶场下载&#xff1a; https://download.vu…

[WASAPI]从Qt MultipleMedia来看WASAPI

[WASAPI] 从Qt MultipleMedia 来看WASAPI 最近在学习有关Windows上的音频驱动相关的知识&#xff0c;在正式开始说WASAPI之前&#xff0c;我想先说一说Qt的Multiple Media&#xff0c;为什么呢&#xff1f;因为Qt的MultipleMedia实际上是WASAPI的一层封装&#xff0c;它在是线…

Linux下编译安装Kokkos

本文记录在Linux下编译安装Kokkos的流程。 零、环境 操作系统Ubuntu 22.04.4 LTSVS Code1.92.1Git2.34.1GCC11.4.0CMake3.22.1oneAPI2024.2.1 一、安装依赖 二、编译安装 参考文献 Mills R T. PETSc/TAO Developments for Early Exascale Systems[J]. 2024.Josef R. A Stud…

HTMLCSS:惊!3D 折叠按钮

这段代码创建了一个具有 3D 效果和动画的按钮&#xff0c;按钮上有 SVG 图标和文本。按钮在鼠标悬停时会显示一个漂浮点动画&#xff0c;图标会消失并显示一个线条动画。这种效果适用于吸引用户注意并提供视觉反馈。按钮的折叠效果和背景渐变增加了页面的美观性。 演示效果 HT…

容器技术所涉及Linux内核关键技术

容器技术所涉及Linux内核关键技术 一、容器技术前世今生 1.1 1979年 — chroot 容器技术的概念可以追溯到1979年的UNIX chroot。它是一套“UNIX操作系统”系统&#xff0c;旨在将其root目录及其它子目录变更至文件系统内的新位置&#xff0c;且只接受特定进程的访问。这项功…

Git远程仓库的多人协作

目录 一.项目克隆 二.多人协作 1.创建林冲仓库 2.协作处理 3.处理冲突 三.分支推送协作 四.分支拉取协作 五.远程分支的删除 一.项目克隆 我们可以把远程项目克隆到本地形成一个本地的仓库 git clone https://github.com/txjava-teach/txjava-code.git //链接你自己的远…