经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » 程序设计 » Python » 查看文章
Python实现上课点名器系统
来源:jb51  时间:2021/10/8 13:08:30  对本文有异议

前言

前段时间,用PyQt5写了几篇文章,关于Python自制一款炫酷音乐播放器、自定义动态壁纸、车牌自动识别系统。今天就继续给大家分享一个读者粉丝投稿的,关于上课点名的实战案例,一起来看看是如何实现的吧!

首先我们还是一起先来看看点名器实现的效果:

在这里插入图片描述

下面,我们开始介绍这款点名器的制作过程。

直接跳到文末获取源码及exe打包程序。

一、核心功能设计

总体来说,我们这款点名器实现的思路大致是,可以自定义设置班级学生姓名或者默认通过学号进行学生随机点名抽取,可以每次单人抽取也可以自己选择连抽人数进行多人连抽,并将这些随机抽取的学生姓名历史数据显示。

拆解需求,接下来我们可以通过以下几步进行实现:

1.UI排版布局设计,确认点名器的各个功能设计

2.读取班级学生的姓名文件,将各个学生的姓名加载到列表中,如果没有学生姓名文件默认创建学号文件

3.实现对学生姓名的随机点名抽取功能

4.实现多人连抽模式

5.查看随机抽取的学生姓名历史数据

二、实现步骤

之前有读者反馈说,不知道程序中具体需要哪些模块、包文件,下面我就把程序中用到模块先放出来。主要包括了文件读写,随机数,PyQt5,win32con等。

  1. import sys,os
  2. from random import randint
  3. from PyQt5 import QtCore, QtGui, QtWidgets
  4. from PyQt5.QtGui import *
  5. from PyQt5.QtCore import Qt, QTimer
  6. from PyQt5.QtWidgets import *
  7. from win32api import MessageBox
  8. from win32con import MB_OK, MB_ICONWARNING

1. UI排版布局设计

根据点名器所需要的功能,首先我们可以进行UI布局设计,我们这次还是使用的pyqt5。主要包含了姓名的随机抽取显示功能、连抽模式的随机抽取显示、抽取历史结果的查看功能、姓名文件的读取显示等。核心设计代码如下:

  1. # author:Dragon少年
  2. def setupUi(self, MainWindow):
  3. #以下课直接粘贴生成的setupui代码
  4. MainWindow.setObjectName("点名器")
  5. MainWindow.resize(420, 360)
  6. self.centralwidget = QtWidgets.QWidget(MainWindow)
  7. self.centralwidget.setObjectName("centralwidget")
  8. self.label = QtWidgets.QLabel(self.centralwidget)
  9. self.label.setGeometry(QtCore.QRect(55, 50, 331, 71))
  10. font = QtGui.QFont()
  11. font.setFamily("宋体")
  12. font.setPointSize(50)
  13. self.label.setFont(font)
  14. self.label.setObjectName("label")
  15. self.pushButton = QtWidgets.QPushButton(self.centralwidget)
  16. self.pushButton.setGeometry(QtCore.QRect(55, 190, 111, 61))
  17. font = QtGui.QFont()
  18. font.setFamily("宋体")
  19. font.setPointSize(20)
  20. self.pushButton.setFont(font)
  21. self.pushButton.setObjectName("pushButton")
  22. self.pushButton_2 = QtWidgets.QPushButton(self.centralwidget)
  23. self.pushButton_2.setGeometry(QtCore.QRect(253, 190, 111, 61))
  24. font = QtGui.QFont()
  25. font.setFamily("宋体")
  26. font.setPointSize(20)
  27. self.pushButton_2.setFont(font)
  28. self.pushButton_2.setObjectName("pushButton_2")
  29. self.pushButton_3 = QtWidgets.QPushButton(self.centralwidget)
  30. self.pushButton_3.setGeometry(QtCore.QRect(11, 570, 111, 41))
  31. self.pushButton_3.setObjectName("pushButton_3")
  32. self.pushButton_4 = QtWidgets.QPushButton(self.centralwidget)
  33. self.pushButton_4.setGeometry(QtCore.QRect(0, 830, 111, 41))
  34. self.pushButton_4.setObjectName("pushButton_4")
  35. self.listWidget = QtWidgets.QListWidget(self.centralwidget)
  36. self.listWidget.setGeometry(QtCore.QRect(11, 370, 397, 191))
  37. self.listWidget.setObjectName("listWidget")
  38. self.label_2 = QtWidgets.QLabel(self.centralwidget)
  39. self.label_2.setGeometry(QtCore.QRect(11, 340, 210, 21))
  40. self.label_2.setObjectName("label_2")
  41. self.pushButton_5 = QtWidgets.QPushButton(self.centralwidget)
  42. self.pushButton_5.setGeometry(QtCore.QRect(11, 303, 111, 20))
  43. self.pushButton_5.setObjectName("pushButton_5")
  44. self.pushButton_6 = QtWidgets.QPushButton(self.centralwidget)
  45. self.pushButton_6.setGeometry(QtCore.QRect(319, 300, 75, 20))
  46. self.pushButton_6.setObjectName("pushButton_6")
  47. self.label_3 = QtWidgets.QLabel(self.centralwidget)
  48. self.label_3.setGeometry(QtCore.QRect(495, 260, 56, 21))
  49. self.label_3.setObjectName("label_3")
  50. self.label_3.setStyleSheet('color:white;background:#222225')
  51. self.pushButton_7 = QtWidgets.QPushButton(self.centralwidget)
  52. self.pushButton_7.setGeometry(QtCore.QRect(649, 240, 111, 61))
  53. font = QtGui.QFont()
  54. font.setFamily("宋体")
  55. font.setPointSize(30)
  56. self.pushButton_7.setFont(font)
  57. self.pushButton_7.setObjectName("pushButton_7")
  58. self.listWidget_2 = QtWidgets.QListWidget(self.centralwidget)
  59. self.listWidget_2.setGeometry(QtCore.QRect(473, 20, 353, 221))
  60. font = QtGui.QFont()
  61. font.setPointSize(14)
  62. self.listWidget_2.setFont(font)
  63. self.listWidget_2.setFocusPolicy(QtCore.Qt.WheelFocus)
  64. self.listWidget_2.setVerticalScrollBarPolicy(QtCore.Qt.ScrollBarAsNeeded)
  65. self.listWidget_2.setHorizontalScrollBarPolicy(QtCore.Qt.ScrollBarAsNeeded) self.listWidget_2.setSizeAdjustPolicy(QtWidgets.QAbstractScrollArea.AdjustToContents)
  66. self.listWidget_2.setObjectName("listWidget_2")
  67. MainWindow.setCentralWidget(self.centralwidget)
  68. self.menubar = QtWidgets.QMenuBar(MainWindow)
  69. self.menubar.setGeometry(QtCore.QRect(0, 0, 874, 23))
  70. self.menubar.setObjectName("menubar")
  71. MainWindow.setMenuBar(self.menubar)
  72. self.statusbar = QtWidgets.QStatusBar(MainWindow)
  73. self.statusbar.setObjectName("statusbar")
  74. MainWindow.setStatusBar(self.statusbar)
  75. self.retranslateUi(MainWindow)
  76. QtCore.QMetaObject.connectSlotsByName(MainWindow)
  77. self.pushButton.clicked.connect(self.start)
  78. self.pushButton_2.clicked.connect(self.stop)
  79. self.pushButton_5.clicked.connect(self.showHistory)
  80. self.pushButton_6.clicked.connect(self.showContinue)
  81. self.pushButton_7.clicked.connect(self.ten)
  82. self.listWidget.setStyleSheet(self.scc)
  83. self.listWidget_2.setStyleSheet(self.scc)
  84. MainWindow.setWindowOpacity(0.95) # 设置窗口透明度
  85. MainWindow.setAttribute(Qt.WA_TranslucentBackground)
  86. MainWindow.setWindowFlag(Qt.FramelessWindowHint) # 隐藏边框
  87.  
  88. self.pushButton_8 = QtWidgets.QPushButton(self.centralwidget)
  89. self.pushButton_8.setGeometry(QtCore.QRect(132, 570, 100, 41))
  90. self.pushButton_8.setObjectName("pushButton_8")
  91. self.pushButton_8.clicked.connect(self.rename)
  92. self.pushButton_8.setText('重置名字文件')
  93.  
  94. # author:Dragon少年
  95. def retranslateUi(self, MainWindow):
  96. self.wide = 420
  97. self.high = 360
  98. _translate = QtCore.QCoreApplication.translate
  99. _translate = QtCore.QCoreApplication.translate
  100. MainWindow.setWindowTitle(_translate("MainWindow", "MainWindow"))
  101. self.label.setText(_translate("MainWindow", "恭喜{}号"))
  102. self.label.setStyleSheet('color:white')
  103. self.pushButton.setText(_translate("MainWindow", "开始"))
  104. self.pushButton_2.setText(_translate("MainWindow", "结束"))
  105. self.pushButton_3.setText(_translate("MainWindow", "打开名字文件"))
  106. self.pushButton_4.setText(_translate("MainWindow", "开gua选项"))
  107. self.label_2.setText(_translate("MainWindow", "点过的学号/姓名:"))
  108. self.pushButton_5.setText(_translate("MainWindow", "查看点过的名字"))
  109. self.pushButton_6.setText(_translate("MainWindow", "连抽模式"))
  110. self.label_3.setText(_translate("MainWindow", "连抽人数"))
  111. self.pushButton_7.setText(_translate("MainWindow", "开始"))

UI实现效果如下:

在这里插入图片描述

对于这个点名器界面,之前有读者粉丝私信我,如何实现将某些内容显示及隐藏动态控制,今天我们就通过这个点名器进行演示实现。

例如对于这个点名器,我们想默认情况下,不要显示抽取的学生历史数据,也不要显示连抽相关的内容,想让这个点名器界面更加简洁直观一些,如下图所示。

在这里插入图片描述

那么这个功能该如何实现呢?其实也非常简单,只要我们定义一个事件函数,通过对界面窗体的宽高进行resize重置,然后将该事件函数绑定到类似按钮点击事件上,就可以实现动态显示/隐藏部分界面内容了。核心代码如下:

  1. # author:Dragon少年
  2. # 点击“查看点过的名字”时,调用该函数,对界面窗体高度大小进行resize重置
  3. def showHistory(self):
  4. global seed
  5. if not seed:
  6. self.high = 656
  7. MainWindow.resize(self.wide,self.high)
  8. seed = True
  9. else:
  10. self.high = 360
  11. MainWindow.resize(self.wide, self.high)
  12. seed = False
  13.  
  14. # 点击“连抽模式”时,调用该函数,对界面窗体宽度大小进行resize重置
  15. def showContinue(self):
  16. global choud
  17. if not choud:
  18. self.wide = 874
  19. MainWindow.resize(self.wide, self.high)
  20. choud = True
  21. else:
  22. self.wide = 420
  23. MainWindow.resize(self.wide, self.high)
  24. choud = False

UI布局设计完成,下面我们开始进行班级学生姓名的读取加载。

2. 学生姓名加载

接下来我们需要对要随机点名抽取的学生姓名进行加载放到一个列表中,这里我们需要注意是否有name.txt文件,如果有可以直接读取加载姓名数据,如果没有该文件,则需要默认根据数字进行学号文件创建。核心代码如下:

  1. # author:Dragon少年
  2. def name():
  3. with open('name.txt', 'w') as f:
  4. print(f.truncate())
  5. print(f.write(a))
  6.  
  7. try:
  8. wordlist3 = []
  9. with open('name.txt', encoding='utf8') as f:
  10. for line in f.readlines():
  11. wordlist3.append(line.strip('\n')) # strip('\n')去掉字符串中的'\n'
  12. print(wordlist3)
  13. name_list = wordlist3
  14. except:
  15. name()
  16. MessageBox(0, "请及时修改当前目录下name文件,默认将为1-52", "MessageBox", MB_OK | MB_ICONWARNING)
  17. name_list = ['1', '2', '3', '4', '5', '6', '7', '8', '9', '10', '11', '12', '13', '14', '15', '16', '17', '18',
  18. '19', '20', '21', '22', '23', '24', '25', '26', '27', '28', '29', '30', '31', '32', '33', '34', '35',
  19. '36', '37', '38', '39', '40', '41', '42', '43', '44', '45', '46', '47', '48', '49', '50', '51', '52', ]
  20.  

3. 随机点名实现

随机点名我们可以设计两个按钮,分别来控制开始抽取和停止抽取功能,通过定时器来进行学生姓名列表数据的随机抽取,从而实现随机点名的功能。核心代码如下:

  1. # author:Dragon少年
  2. # 对姓名列表进行随机显示 模拟随机点名功能
  3. def setname(self):
  4. global running
  5. global name
  6. try:
  7. name = name_list[randint(0, len(name_list) - 1)]
  8. self.label.setText("恭喜{}号!!!".format(name))
  9. except:
  10. self.name()
  11. reply = QtWidgets.QMessageBox.warning(self, u'警告', u'发生错误,请检查name文件的学号后再重新打开本软件', QtWidgets.QMessageBox.Yes)
  12. sys.exit()
  13.  
  14. # 开始按钮绑定 通过定时器加载随机点名函数
  15. def start(self):
  16. global running
  17. if running:
  18. print('running')
  19. pass
  20. else:
  21. self.timer = QTimer(self)
  22. self.timer.timeout.connect(self.setname)
  23. self.timer.start(50)
  24. running = 'True'
  25.  
  26. # 结束按钮绑定 控制定时器停止
  27. def stop(self):
  28. global running, a
  29. if running:
  30. self.timer.stop()
  31. running = False
  32. self.listWidget.addItem(name)
  33. else:
  34. reply = QtWidgets.QMessageBox.warning(self, u'警告', u'还没开始就想结束?', QtWidgets.QMessageBox.Yes)

至此,我们已经可以实现点名功能了,如下所示。

在这里插入图片描述

4. 连抽模式实现

随机抽取功能已经实现了,多人连抽和随机抽取类似,只要进行姓名列表随机抽取对应人数就可以了。当然对抽取人数的输入,我们需要进行一些输入限制,避免一些异常输入情况,例如输入非数字、输入负数、输入数字过大等。我们只要将连抽函数同样绑定到按钮上就可以了。核心代码如下:

  1. # author:Dragon少年
  2. def ten(self):
  3. num = self.lineEdit.text()
  4. print (num)
  5. num = int(num)
  6. if not num =='' and not num<=0 and not num>1000:
  7. if num > 20:
  8. reply = QtWidgets.QMessageBox.warning(self, u'警告', u'认真的吗,这么多', QtWidgets.QMessageBox.Yes)
  9. self.listWidget_2.clear()
  10. for i in range (0,int(num)):
  11. name = name_list[randint(0, len(name_list) - 1)]
  12. self.listWidget_2.addItem(name)
  13. self.listWidget.addItem(name)
  14. elif num =='':
  15. reply = QtWidgets.QMessageBox.warning(self, u'警告', u'请输入数字', QtWidgets.QMessageBox.Yes)
  16. self.listWidget_2.clear()
  17. elif num<0:
  18. #win32api.MessageBox(0, "你见过负数个人么???????", "通知", win32con.MB_OK | win32con.MB_ICONWARNING)
  19. reply = QtWidgets.QMessageBox.warning(self, u'警告', u'人数负数,输入有误!', QtWidgets.QMessageBox.Yes)
  20. self.listWidget_2.clear()
  21. elif num==0:
  22. #win32api.MessageBox(0, "人都被你吃了???", "通知", win32con.MB_OK | win32con.MB_ICONWARNING)
  23. reply = QtWidgets.QMessageBox.warning(self, u'警告', u'人数为0,输入有误!', QtWidgets.QMessageBox.Yes)
  24. self.listWidget_2.clear()
  25. elif num>1000:
  26. #win32api.MessageBox(0, "这么大?要不起~", "通知", win32con.MB_OK | win32con.MB_ICONWARNING)
  27. reply = QtWidgets.QMessageBox.warning(self, u'警告', u'人数超出限制,输入有误!', QtWidgets.QMessageBox.Yes)
  28. self.listWidget_2.clear()

5. 抽取历史查看

我们最后再实现一个对点名的学生姓名进行历史数据显示查看功能。该功能实现也很简单,只要对上面随机抽取的姓名添加到对应的历史列表框中就可以了;还可以对学生姓名文件进行重置等功能。核心代码如下:

  1. # author:Dragon少年
  2. def rename(self):
  3. reply = QtWidgets.QMessageBox.question(self, u'警告', u'确定重置name文件为1-52?', QtWidgets.QMessageBox.Yes,
  4. QtWidgets.QMessageBox.No)
  5. if reply == QtWidgets.QMessageBox.Yes:
  6. with open('name.txt', 'w') as f:
  7. print(f.truncate())
  8. print(f.write(a))
  9. MessageBox(0, "重置完成,", "通知", MB_OK | MB_ICONWARNING)
  10. else:
  11. pass

至此,这款点名器就完成了, 一起来看看效果吧!

在这里插入图片描述

三. 打包配置

最近有好几个粉丝问我,Python能不能打包成exe?如何打包呢?之前已经介绍过了,今天就通过这个再来来一起介绍下。

Python常用的打包工具是第三方库Pyinstaller,需要安装下pyinstaller。

  1. pip install Pyinstaller

接下来,我们需要打开命令窗口,切换到项目目录下再执行打包命令。

  1. pyinstaller -F -i ./img.ico 点名.py

打包常用的参数如下:

  • -F 表示生成单个可执行文件
  • -p 表示你自己自定义需要加载的类路径,一般情况下用不到
  • -i 表示可执行文件的图标
  • -w 表示去掉控制台窗口,这在GUI界面时非常有用

之前有小伙伴问我,打包的图标需要.ico后缀的图片,应该如何把普通图片转成图标格式.ico呢?Python也可以帮你实现图片转.ico文件。核心代码如下:

  1. import PythonMagick
  2.  
  3. # 生成图标ico(png格式图片转成ico)
  4. img = PythonMagick.Image('./image/img1/1.png')
  5. # 这里要设置一下尺寸,不然会报ico尺寸异常错误
  6. img.sample('128x128')
  7. img.write('./img.ico')

总结

本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注w3xue的更多内容!

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号