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é?
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_())