python - 线程实时记录

标签 python multithreading pyqt pyqt5 qthread

我正在编写一个用于调整照片大小的简单脚本。我想要一个带有文本字段的小部件,其中在调整每个文件的大小后会显示消息。

#!/usr/bin/python3
# -*- coding: utf-8 -*-

import time, sys
from PyQt5.QtCore import pyqtSignal, QThread
from PyQt5.QtWidgets import QApplication, QPushButton, QTextEdit, QWidget, QVBoxLayout

class Thread(QThread):
    log = pyqtSignal(str)
    def __init__(self, parent=None):
        super(Thread, self).__init__(parent)
    def test(self, i):
        time.sleep(1)
        self.log.emit(str(i))

class Widget(QWidget):
    def __init__(self):
        super().__init__()
        self.ui()
    def process(self):
        self.toLog('some text...')
        worker = Thread()
        worker.log.connect(self.toLog)
        for i in range(1, 5):
            worker.test(i)
    def ui(self):
        self.LogOutputTxt = QTextEdit()
        self.LogOutputTxt.setReadOnly(True)
        startBtn = QPushButton('Start')
        startBtn.clicked.connect(self.start)
        layout = QVBoxLayout()
        layout.addWidget(self.LogOutputTxt)
        layout.addWidget(startBtn)
        self.setLayout(layout)
        self.resize(400, 300)
        self.show()
    def start(self):
        self.toLog('start')
        self.process()
    def toLog(self, txt):
        self.LogOutputTxt.append(txt)

if __name__ == '__main__':
    app = QApplication(sys.argv)
    ui = Widget()
    sys.exit(app.exec_())

到目前为止,在调整所有文件的大小后,所有消息都会立即出现。有什么方法可以一一完成(我的意思是调整文件大小、显示消息等)?

最佳答案

下面是重写的脚本,应该符合您的要求。

但请注意,这非常简单,并没有为确保线程安全而努力。 setItems 方法只是对传递给它的数据进行浅拷贝——这只适用于不可变对象(immutable对象)列表。您还必须确保您永远不会在工作线程内执行任何 gui 操作,其中包括对像素图的操作。如果要操作图像,请使用 QImage。 (如果您想知道如何停止正在运行的线程,请参阅示例 this answer)。

#!/usr/bin/python3
# -*- coding: utf-8 -*-

import time, sys
from PyQt5.QtCore import pyqtSignal, QThread
from PyQt5.QtWidgets import (
    QApplication, QPushButton, QTextEdit, QWidget, QVBoxLayout
    )

class Thread(QThread):
    log = pyqtSignal(str)

    def __init__(self, parent=None):
        super(Thread, self).__init__(parent)
        self._items = []

    def setItems(self, items):
        if not self.isRunning():
            self._items[:] = items

    def run(self):
        for item in self._items:
            time.sleep(1)
            self.log.emit('processing: %s' % item)

class Widget(QWidget):
    def __init__(self):
        super().__init__()
        self.ui()
        self._worker = Thread(self)
        self._worker.log.connect(self.toLog)
        self._worker.started.connect(lambda: self.toLog('start'))
        self._worker.finished.connect(lambda: self.toLog('finished'))

    def process(self):
        items = ['Image%02d.png' % i for i in range(10)]
        self._worker.setItems(items)
        self._worker.start()

    def ui(self):
        self.LogOutputTxt = QTextEdit()
        self.LogOutputTxt.setReadOnly(True)
        startBtn = QPushButton('Start')
        startBtn.clicked.connect(self.start)
        layout = QVBoxLayout()
        layout.addWidget(self.LogOutputTxt)
        layout.addWidget(startBtn)
        self.setLayout(layout)
        self.resize(400, 300)
        self.show()

    def start(self):
        if not self._worker.isRunning():
            self.process()

    def toLog(self, txt):
        self.LogOutputTxt.append(txt)

if __name__ == '__main__':

    app = QApplication(sys.argv)
    ui = Widget()
    sys.exit(app.exec_())

关于python - 线程实时记录,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/40575455/

相关文章:

python - Pandas 聚合有序点坐标

python - 使用 Python 在后台下载文件

c# - 如何从另一个线程将项目添加到 ListView 而不会导致异常

python - 我应该在我的 PyQt 应用程序中使用 `app.exec()` 还是 `app.exec_()`?

python - 使用 Boto 查找安装了哪个设备和 EBS 卷

python - Polars 连接器 x SQL Server 的 Windows 身份验证

c++ - 线程安全管道端接

Ruby:如何将多线程合并到这个网络抓取场景中?

python - pylint 错误在 python3.2 中运行正常,但在 python3.3 中失败

python - QGraphicsGridLayout 适用于 PySide 但不适用于 PyQt4