pyqt5如何计时?

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了pyqt5如何计时?相关的知识,希望对你有一定的参考价值。

就是类似一个计时器,计时几分几秒后做某事

参考技术A 利用计时器模块QTimer#初始化一个定时器s追问

可是qtime不可以设定计时的时间

本回答被提问者采纳
参考技术B 用python实现计时器功能,代码如下:''' Simple Timing Function.This function prints out a message with the elaps追问

注意是pyqt5

从不同类添加的小部件内的 PyQt5 停止计时器

【中文标题】从不同类添加的小部件内的 PyQt5 停止计时器【英文标题】:PyQt5 stop timer inside a widget that was added from a different class 【发布时间】:2020-01-25 12:53:04 【问题描述】:

我想在多个窗口的顶部显示当前日期和时间,所以我创建了这个顶部小部件的一个类。它可以工作,但是当我切换到另一个窗口时,计时器也会继续在前一个窗口上运行。如何在切换窗口之前停止计时器,也许让相同的实例在新窗口上运行?

from PyQt5 import QtCore
from PyQt5 import QtWidgets
from PyQt5.QtWidgets import QApplication, QMainWindow, QHBoxLayout, QWidget
import sys
from datetime import datetime

CurrentWindow = None

class TopBar(QWidget):
    def __init__(self):
        super().__init__()
        self.initUI()

    def initUI(self):
        self.labelTime = QtWidgets.QLabel(self)
        self.labelTime.setStyleSheet("background-color: rgba(0, 0, 0, 0); color: white")

        background = QtWidgets.QWidget(self)
        background.setStyleSheet("background-color: rgba(0, 191, 255, 0.6)")
        background.setGeometry(0, 0, 480, 30)

        hbox = QHBoxLayout(background)
        hbox.setContentsMargins(10, 0, 10, 0)
        hbox.addWidget(self.labelTime, alignment=QtCore.Qt.AlignRight)

        self.timer = QtCore.QTimer(self)
        self.timer.setInterval(1000)
        self.timer.timeout.connect(self.displayTime)
        self.timer.start()

        self.displayTime()


    def displayTime(self):
        print(self.parent())
        self.labelTime.setText(datetime.now().strftime("%Y/%m/%d %H:%M:%S"))


class Window1(QMainWindow):
    def __init__(self):
        super().__init__()
        self.initUI()

    def initUI(self):
        self.resize(480, 320)

        self.centralwidget = QtWidgets.QWidget(self)
        self.setCentralWidget(self.centralwidget)

        widgetTop = QtWidgets.QWidget(self.centralwidget)
        widgetTop.setGeometry(0, 0, 480, 30)

        layoutTop = QHBoxLayout(widgetTop)
        layoutTop.addWidget(TopBar())
        layoutTop.setContentsMargins(0, 0, 0, 0)

        self.pushButton = QtWidgets.QPushButton(self.centralwidget)
        self.pushButton.setText('Go to Window2')
        self.pushButton.clicked.connect(self.goToWindow2)

        layoutCenter = QHBoxLayout(self.centralwidget)
        layoutCenter.addWidget(self.pushButton, alignment=QtCore.Qt.AlignCenter)

        self.show()


    def goToWindow2(self):
        global CurrentWindow
        CurrentWindow = Window2()
        self.close()


class Window2(QMainWindow):
    def __init__(self):
        super().__init__()
        self.initUI()

    def initUI(self):
        self.resize(480, 320)

        self.centralwidget = QtWidgets.QWidget(self)
        self.setCentralWidget(self.centralwidget)

        widgetTop = QtWidgets.QWidget(self.centralwidget)
        widgetTop.setGeometry(0, 0, 480, 30)

        layoutTop = QHBoxLayout(widgetTop)
        layoutTop.addWidget(TopBar())
        layoutTop.setContentsMargins(0, 0, 0, 0)

        self.pushButton = QtWidgets.QPushButton(self.centralwidget)
        self.pushButton.setText('Go to Window1')
        self.pushButton.clicked.connect(self.goToWindow1)

        layoutCenter = QHBoxLayout(self.centralwidget)
        layoutCenter.addWidget(self.pushButton, alignment=QtCore.Qt.AlignCenter)

        self.show()


    def goToWindow1(self):
        global CurrentWindow
        CurrentWindow = Window1()
        self.close()



if __name__=='__main__':
    app = QApplication(sys.argv)
    ex = Window1()
    sys.exit(app.exec_())

【问题讨论】:

【参考方案1】:

我不认为 QTimer 的执行会减慢应用程序的速度。

但我仍然会向您展示如何从另一个窗口阻止它。为此,您必须访问该对象,因此您必须成为该类的成员,然后当您更改窗口时,您可以使用 QTimer 的 stop 方法停止它。

class Window1(QMainWindow):
    # ...

    def initUI(self):
        # ...
        layoutTop = QHBoxLayout(widgetTop)
        self.topbar = TopBar()
        layoutTop.addWidget(self.topbar)
        layoutTop.setContentsMargins(0, 0, 0, 0)
        # ...

    def goToWindow2(self):
        global CurrentWindow
        self.topbar.timer.stop()
        CurrentWindow = Window2()
        self.close()


class Window2(QMainWindow):
    # ...

    def initUI(self):
        # ...
        layoutTop = QHBoxLayout(widgetTop)
        self.topbar = TopBar()
        layoutTop.addWidget(self.topbar)
        layoutTop.setContentsMargins(0, 0, 0, 0)
        # ...

    def goToWindow1(self):
        global CurrentWindow
        self.topbar.timer.stop()
        CurrentWindow = Window1()
        self.close()

如果您仍然认为错误的原因是有多个 QTimer,那么在下面的代码中将只有一个 TopBar,并且他们将使用 QStackedWidget 更改小部件

from PyQt5 import QtCore
from PyQt5 import QtWidgets
from PyQt5.QtWidgets import QApplication, QMainWindow, QHBoxLayout, QWidget
import sys
from datetime import datetime


class TopBar(QWidget):
    def __init__(self):
        super().__init__()
        self.initUI()

    def initUI(self):
        self.setAttribute(QtCore.Qt.WA_StyledBackground)
        self.labelTime = QtWidgets.QLabel()
        self.labelTime.setStyleSheet("background-color: rgba(0, 0, 0, 0); color: white")

        self.setStyleSheet("background-color: rgba(0, 191, 255, 0.6)")
        self.setFixedHeight(30)

        hbox = QHBoxLayout(self)
        hbox.setContentsMargins(10, 0, 10, 0)
        hbox.addWidget(self.labelTime, alignment=QtCore.Qt.AlignRight)

        self.timer = QtCore.QTimer(self)
        self.timer.setInterval(1000)
        self.timer.timeout.connect(self.displayTime)
        self.timer.start()

        self.displayTime()

    def displayTime(self):
        self.labelTime.setText(datetime.now().strftime("%Y/%m/%d %H:%M:%S"))


class Window(QWidget):
    changeWindow = QtCore.pyqtSignal(int)

    def changeTo(self, index):
        def callback():
            self.changeWindow.emit(index)

        return callback


class Window1(Window):
    def __init__(self):
        super().__init__()
        self.initUI()

    def initUI(self):
        self.pushButton = QtWidgets.QPushButton()
        self.pushButton.setText("Go to Window2")
        self.pushButton.clicked.connect(self.changeTo(1))

        layoutCenter = QHBoxLayout(self)
        layoutCenter.addWidget(self.pushButton, alignment=QtCore.Qt.AlignCenter)


class Window2(Window):
    def __init__(self):
        super().__init__()
        self.initUI()

    def initUI(self):
        self.pushButton = QtWidgets.QPushButton()
        self.pushButton.setText("Go to Window1")
        self.pushButton.clicked.connect(self.changeTo(0))

        layoutCenter = QHBoxLayout(self)
        layoutCenter.addWidget(self.pushButton, alignment=QtCore.Qt.AlignCenter)


class MainWindow(QtWidgets.QMainWindow):
    def __init__(self, parent=None):
        super().__init__(parent)
        self.initUI()

    def initUI(self):
        self.resize(480, 320)

        self.centralwidget = QtWidgets.QWidget(self)
        self.setCentralWidget(self.centralwidget)

        self.topbar = TopBar()
        lay = QtWidgets.QVBoxLayout(self.centralwidget)
        lay.setContentsMargins(0, 0, 0, 0)
        lay.addWidget(self.topbar)

        stacked_widget = QtWidgets.QStackedWidget()
        lay.addWidget(stacked_widget)

        for w in (Window1(), Window2()):
            stacked_widget.addWidget(w)
            if isinstance(w, Window):
                w.changeWindow.connect(stacked_widget.setCurrentIndex)


if __name__ == "__main__":
    app = QApplication(sys.argv)
    w = MainWindow()
    w.show()
    sys.exit(app.exec_())

【讨论】:

第二个选项对我来说似乎是最佳解决方案,谢谢!我有一个连接到计时器的耗时函数,这会减慢应用程序的速度,但我用 QThread 替换了它,它也应该只有一个带有第二个选项的实例。 @SmileXS4 所以问题不是 QTimer 而是耗时的函数,你不这么认为吗?在您的问题中,没有任何内容。如果你有一个耗时的任务,那么你不应该在主线程中执行它,而是在辅助线程中执行它。如果您需要帮助,您必须提供真实的minimal reproducible example,否则您会混淆社区。​​span> 抱歉,我编辑了这个问题,因为我需要帮助来一次只运行一个计时器,而不是更快地运行应用程序。

以上是关于pyqt5如何计时?的主要内容,如果未能解决你的问题,请参考以下文章

PyQt5 应用程序 - 使用计时器运行带有循环的函数使其“冻结”

PyQt5 QThread 不会因终止或标志变量而停止

第九篇 -- 定时器和计时器

为啥按下 PyQt 按钮时 matplotlib 的计时器会加快?

PyQt5:最小化窗口后 QTimer 不同步(OSX)

c#计时器如何保存计时