Aktualizacja elementów GUI w wielowątkowym PyQT
Przez jakiś czas szukałem informacji Jak zrobić program wielowątkowy za pomocą PyQT, aktualizując GUI, aby pokazać wyniki.
Jestem przyzwyczajony do nauki na przykładzie i nie mogę znaleźć (tak szukałem od tygodni) żadnego prostego przykładu programu wykorzystującego wielowątkowość wykonującego tak proste zadanie, jak na przykład łączenie się z listą stron www (5 wątków) i drukowanie przetworzonych adresów URL z kodem odpowiedzi.
Czy ktoś mógłby udostępnić kod lub wysłać mnie do dobrego tutoriala gdzie taki program jest wyjaśniony ?
1 answers
Oto kilka bardzo podstawowych przykładów.
Odniesienia do elementów GUI można przekazywać do wątków i aktualizować je w wątku.
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_())
Uwaga redaktorów: widżety Qt nie są bezpieczne dla wątków i nie powinny być dostępne z żadnego wątku poza głównym wątkiem(zobacz dokumentacja Qt aby uzyskać więcej szczegółów). Prawidłowym sposobem korzystania z wątków są sygnały/sloty, jak pokazuje druga część tej odpowiedzi.
Również, można użyć sygnałów i slotów, aby oddzielić gui i logiki sieci.
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_())
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2015-07-27 05:15:39