web-dev-qa-db-fra.com

Mise à jour des éléments de l'interface graphique dans MultiThreaded PyQT

Je cherchais depuis un certain temps pour trouver des informations comment faire un programme multithread en utilisant PyQT, mettre à jour l'interface graphique pour afficher les résultats.

J'ai l'habitude d'apprendre par l'exemple et je ne trouve pas (oui je cherchais depuis des semaines) un exemple simple de programme utilisant le multithreading faisant une tâche aussi simple que par exemple la connexion à la liste des sites www (5 threads) et juste l'impression traitée URL avec code de réponse.

Quelqu'un pourrait-il partager du code ou m'envoyer à un bon tutoriel où un tel programme est expliqué?

19
Nuncjo

Voici quelques exemples très basiques.

Vous pouvez transmettre des références aux éléments d'interface graphique aux threads et les mettre à jour dans thread.

import sys
import urllib2

from PyQt4 import QtCore, QtGui


class DownloadThread(QtCore.QThread):
    def __init__(self, url, list_widget):
        QtCore.QThread.__init__(self)
        self.url = url
        self.list_widget = list_widget

    def run(self):
        info = urllib2.urlopen(self.url).info()
        self.list_widget.addItem('%s\n%s' % (self.url, info))


class MainWindow(QtGui.QWidget):
    def __init__(self):
        super(MainWindow, self).__init__()
        self.list_widget = QtGui.QListWidget()
        self.button = QtGui.QPushButton("Start")
        self.button.clicked.connect(self.start_download)
        layout = QtGui.QVBoxLayout()
        layout.addWidget(self.button)
        layout.addWidget(self.list_widget)
        self.setLayout(layout)

    def start_download(self):
        urls = ['http://google.com', 'http://Twitter.com', 'http://yandex.ru',
                'http://stackoverflow.com/', 'http://www.youtube.com/']
        self.threads = []
        for url in urls:
            downloader = DownloadThread(url, self.list_widget)
            self.threads.append(downloader)
            downloader.start()

if __name__ == "__main__":
    app = QtGui.QApplication(sys.argv)
    window = MainWindow()
    window.resize(640, 480)
    window.show()
    sys.exit(app.exec_())

Note des éditeurs: Les widgets Qt ne sont pas thread-safe et ne doivent pas être accessibles depuis n'importe quel thread sauf le thread principal (voir la documentation Qt = pour plus de détails). La bonne façon d'utiliser les threads est via des signaux/slots comme le montre la deuxième partie de cette réponse.


En outre, vous pouvez utiliser des signaux et des slots pour séparer l'interface graphique et la logique réseau.

import sys
import urllib2

from PyQt4 import QtCore, QtGui


class DownloadThread(QtCore.QThread):

    data_downloaded = QtCore.pyqtSignal(object)

    def __init__(self, url):
        QtCore.QThread.__init__(self)
        self.url = url

    def run(self):
        info = urllib2.urlopen(self.url).info()
        self.data_downloaded.emit('%s\n%s' % (self.url, info))


class MainWindow(QtGui.QWidget):
    def __init__(self):
        super(MainWindow, self).__init__()
        self.list_widget = QtGui.QListWidget()
        self.button = QtGui.QPushButton("Start")
        self.button.clicked.connect(self.start_download)
        layout = QtGui.QVBoxLayout()
        layout.addWidget(self.button)
        layout.addWidget(self.list_widget)
        self.setLayout(layout)

    def start_download(self):
        urls = ['http://google.com', 'http://Twitter.com', 'http://yandex.ru',
                'http://stackoverflow.com/', 'http://www.youtube.com/']
        self.threads = []
        for url in urls:
            downloader = DownloadThread(url)
            downloader.data_downloaded.connect(self.on_data_ready)
            self.threads.append(downloader)
            downloader.start()

    def on_data_ready(self, data):
        print data
        self.list_widget.addItem(unicode(data))


if __name__ == "__main__":
    app = QtGui.QApplication(sys.argv)
    window = MainWindow()
    window.resize(640, 480)
    window.show()
    sys.exit(app.exec_())
33
reclosedev