我是Python的新手,我制作了一个非常简单的倒计时计时器。GUI是在Qt Designer中创建的。有一个用于输入秒数的旋转框、一个启动按钮和一个LCD数字计数器。计数器使用以下代码倒计时:
def start_btn_clicked(self):
x = self.Minute_spinBox.value()
for i in xrange(x,0,-1):
time.sleep(1)
print (i)
为了在玩它的时候看到发生了什么,我添加了打印指令,这样它就可以在运行时在Python控制台中显示倒计时。然后我想我可以很容易地让LCD数字显示倒计时,比如:
self.lcdNumber.display(i)("%SS")
但无论我做什么,我都无法把它展示出来。对于上面的行,我得到了显示的第一个数字,但随后我得到了一个错误:
self.lcdNumber.display(i)("%SS")
TypeError: 'NoneType' object is not callable
我尝试了太多的变体,以至于我不再知道我从哪里开始,而我认为这会很简单。我很想知道为什么我不能让它显示倒计时。
只需在原始代码中添加一行代码,lcdNumber就可以显示计数。。。
def start_btn_clicked(self):
x = self.Minute_spinBox.value()
for i in xrange(x,0,-1):
time.sleep(1)
app.processEvents() # just this one line allows display of 'i'
self.lcdNumber.display(i)`
并且可以完美地进行
display
函数返回None
,因此显然不允许执行None("%SS")
。
self.lcdNumber.display(i)
足以显示倒计时!
让Qt在循环时绘制小部件,从另一个线程运行倒计时。请参见示例。
import time
from threading import Thread
from PyQt4.QtGui import QApplication, QMainWindow, QLCDNumber
class Window(QMainWindow):
def __init__(self):
QMainWindow.__init__(self)
self.lcdnumber = QLCDNumber(self)
self.resize(400, 400)
t = Thread(target=self._countdown)
t.start()
def _countdown(self):
x = 10
for i in xrange(x,0,-1):
time.sleep(1)
self.lcdnumber.display(i)
if __name__ == "__main__":
app = QApplication([])
window = Window()
window.show()
app.exec_()
for
循环正在阻塞GUI。
连接到按钮clicked
信号的插槽被同步处理。这意味着事件循环必须等待插槽返回,然后才能处理更多事件(包括更新GUI所需的绘制事件)。
因此,您需要找到在for
循环运行时处理这些事件的方法。有多种方法可以做到这一点,例如使用QTimer或QThread。但是,修复特定示例的最简单方法是使用QCoreApplication.prrocessEvents.
这里有一个例子展示了如何做到这一点:
import sys, time
from PyQt4 import QtCore, QtGui
class Window(QtGui.QWidget):
def __init__(self):
super(Window, self).__init__()
self.spinbox = QtGui.QSpinBox(self)
self.spinbox.setValue(5)
self.lcdnumber = QtGui.QLCDNumber(self)
self.button = QtGui.QPushButton('Start', self)
self.button.clicked.connect(self.handleButton)
layout = QtGui.QVBoxLayout(self)
layout.addWidget(self.spinbox)
layout.addWidget(self.lcdnumber)
layout.addWidget(self.button)
def handleButton(self):
for tick in range(self.spinbox.value(), -1, -1):
self.lcdnumber.display(tick)
self.button.setEnabled(not tick)
# continually process events for one second
start = time.time()
while time.time() - start < 1:
QtGui.qApp.processEvents()
time.sleep(0.02)
if __name__ == '__main__':
app = QtGui.QApplication(sys.argv)
window = Window()
window.setGeometry(500, 300, 300, 200)
window.show()
sys.exit(app.exec_())