Как правильно использовать QThread в pyqt с moveToThread()?

я прочитал эту статью, как на самом деле, действительно использовать QThreads; В полном объяснении говорится, что вместо подкласса qthread и переопределения run() необходимо использовать moveToThread, чтобы поместить объект QObject в экземпляр QThread с помощью moveToThread (QThread *)

Вот пример c++, но я не знаю, как преобразовать его в код Python.

class Worker : public QObject
 {
     Q_OBJECT
     QThread workerThread;

 public slots:
     void doWork(const QString &parameter) {
         // ...
         emit resultReady(result);
     }

 signals:
     void resultReady(const QString &result);
 };

 class Controller : public QObject
 {
     Q_OBJECT
     QThread workerThread;
 public:
     Controller() {
         Worker *worker = new Worker;
         worker->moveToThread(&workerThread);
         connect(workerThread, SIGNAL(finished()), worker, SLOT(deleteLater()));
         connect(this, SIGNAL(operate(QString)), worker, SLOT(doWork(QString)));
         connect(worker, SIGNAL(resultReady(QString)), this, SLOT(handleResults(QString)));
         workerThread.start();
     }
     ~Controller() {
         workerThread.quit();
         workerThread.wait();
     }
 public slots:
     void handleResults(const QString &);
 signals:
     void operate(const QString &);
 };



QThread* thread = new QThread;
Worker* worker = new Worker();
worker->moveToThread(thread);
connect(worker, SIGNAL(error(QString)), this, SLOT(errorString(QString)));
connect(thread, SIGNAL(started()), worker, SLOT(process()));
connect(worker, SIGNAL(finished()), thread, SLOT(quit()));
connect(worker, SIGNAL(finished()), worker, SLOT(deleteLater()));
connect(thread, SIGNAL(finished()), thread, SLOT(deleteLater()));
thread->start();

Я использовал этот метод для создания qthread, но, как вы можете видеть, он использует не рекомендуемый способ. Как я могу переписать его, чтобы использовать предпочтительный метод?

class GenericThread(QThread):
    def __init__(self, function, *args, **kwargs):
        QThread.__init__(self)
        # super(GenericThread, self).__init__()

        self.function = function
        self.args = args
        self.kwargs = kwargs

    def __del__(self):
        self.wait()

    def run(self, *args):
        self.function(*self.args, **self.kwargs)

редактировать: два года спустя... Я попробовал Qris 'код, он работает и в другой теме

import sys
import time
from PyQt4 import QtCore, QtGui
from PyQt4.QtCore import pyqtSignal, pyqtSlot
import threading


def logthread(caller):
    print('%-25s: %s, %s,' % (caller, threading.current_thread().name,
                              threading.current_thread().ident))


class MyApp(QtGui.QWidget):

    def __init__(self, parent=None):
        QtGui.QWidget.__init__(self, parent)

        self.setGeometry(300, 300, 280, 600)
        self.setWindowTitle('using threads')

        self.layout = QtGui.QVBoxLayout(self)

        self.testButton = QtGui.QPushButton("QThread")
        self.testButton.released.connect(self.test)
        self.listwidget = QtGui.QListWidget(self)

        self.layout.addWidget(self.testButton)
        self.layout.addWidget(self.listwidget)

        self.threadPool = []
        logthread('mainwin.__init__')

    def add(self, text):
        """ Add item to list widget """
        logthread('mainwin.add')
        self.listwidget.addItem(text)
        self.listwidget.sortItems()

    def addBatch(self, text="test", iters=6, delay=0.3):
        """ Add several items to list widget """
        logthread('mainwin.addBatch')
        for i in range(iters):
            time.sleep(delay)  # artificial time delay
            self.add(text+" "+str(i))

    def test(self):
        my_thread = QtCore.QThread()
        my_thread.start()

        # This causes my_worker.run() to eventually execute in my_thread:
        my_worker = GenericWorker(self.addBatch)
        my_worker.moveToThread(my_thread)
        my_worker.start.emit("hello")
        # my_worker.finished.connect(self.xxx)

        self.threadPool.append(my_thread)
        self.my_worker = my_worker


class GenericWorker(QtCore.QObject):

    start = pyqtSignal(str)
    finished = pyqtSignal()

    def __init__(self, function, *args, **kwargs):
        super(GenericWorker, self).__init__()
        logthread('GenericWorker.__init__')
        self.function = function
        self.args = args
        self.kwargs = kwargs
        self.start.connect(self.run)

    @pyqtSlot()
    def run(self, *args, **kwargs):
        logthread('GenericWorker.run')
        self.function(*self.args, **self.kwargs)
        self.finished.emit()


# run
app = QtGui.QApplication(sys.argv)
test = MyApp()
test.show()
app.exec_()

результат:

mainwin.__init__         : MainThread, 140221684574016,
GenericWorker.__init__   : MainThread, 140221684574016,
GenericWorker.run        : Dummy-1, 140221265458944,
mainwin.addBatch         : Dummy-1, 140221265458944,
mainwin.add              : Dummy-1, 140221265458944,
mainwin.add              : Dummy-1, 140221265458944,
mainwin.add              : Dummy-1, 140221265458944,
mainwin.add              : Dummy-1, 140221265458944,
mainwin.add              : Dummy-1, 140221265458944,
mainwin.add              : Dummy-1, 140221265458944,

Ответ 1

Реализация run() по умолчанию в QThread запускает цикл событий для вас, эквивалент:

class GenericThread(QThread):
    def run(self, *args):
        self.exec_()

Важным элементом цикла событий является то, что он позволяет объектам , принадлежащим потоку получать события на своих слотах, которые будут выполняться в этом потоке. Эти объекты являются только QObjects, а не QThreads.

Важное примечание: объект QThread не принадлежит собственному потоку! Он был создан на главной теме и живет там. Помимо метода запуска, весь его код выполняется в основном потоке.

Итак, вы должны это сделать:

class GenericWorker(QObject):
    def __init__(self, function, *args, **kwargs):
        super(GenericWorker, self).__init__()

        self.function = function
        self.args = args
        self.kwargs = kwargs
        self.start.connect(self.run)

    start = pyqtSignal(str)

    @pyqtSlot
    def run(self, some_string_arg):
        self.function(*self.args, **self.kwargs)

my_thread = QThread()
my_thread.start()

# This causes my_worker.run() to eventually execute in my_thread:
my_worker = GenericWorker(...)
my_worker.moveToThread(my_thread)
my_worker.start.emit("hello")

Также подумайте о том, что происходит с результатом self.function, который в настоящее время отбрасывается. Вы можете объявить другой сигнал на GenericWorker, который получает результат, и метод run() испускает этот сигнал, когда он это делает, передавая ему результат.

Как только вы получите зависание и осознаете, что нет и не должны подкласса QThread, жизнь становится намного проще и проще. Проще говоря, никогда не работайте в QThread. Вам почти не нужно отменять запуск. В большинстве случаев использование правильных ассоциаций с QObject для QThread и использование сигналов/слотов QT создает чрезвычайно мощный способ выполнения многопоточного программирования. Просто будьте осторожны, чтобы позволить QObjects, которые вы нажали на ваши рабочие потоки, повсюду...

http://ilearnstuff.blogspot.co.uk/2012/09/qthread-best-practices-when-qthread.html

Ответ 2

Я пытался использовать пример qris в своем приложении, но продолжал использовать свой код в моем основном потоке! Это тот сигнал, который он объявил, чтобы вызвать бег!

В принципе, когда вы подключаете его в конструкторе объекта, соединение будет существовать между двумя объектами в основном потоке, потому что свойства QObject принадлежат потоку, который их создал. Когда вы перемещаете QObject в новый поток, соединение не перемещается с вами. Уберите линию, которая соединяет ваш сигнал с функцией запуска, и соедините его после перемещения рабочего в новый поток!

Соответствующее изменение ответа qris:

class GenericWorker(QObject):
    def __init__(self, function, *args, **kwargs):
        super(GenericWorker, self).__init__()

        self.function = function
        self.args = args
        self.kwargs = kwargs

    start = pyqtSignal(str)

    @pyqtSlot
    def run(self, some_string_arg):
        self.function(*self.args, **self.kwargs)

my_thread = QThread()
my_thread.start()

# This causes my_worker.run() to eventually execute in my_thread:
my_worker = GenericWorker(...)
my_worker.moveToThread(my_thread)
my_worker.start.connect(my_worker.run) #  <---- Like this instead 
my_worker.start.emit("hello")

Ответ 3

Я пробовал подходы @qris и @MatthewRunchey.

С @pyqtSlot декоратора @pyqtSlot Qt проверяет "местоположение" рабочего экземпляра при moveToThread сигнала: даже если соединение было установлено до того, как moveToThread испустит сигнал после того, как moveToThread выполнит слот в рабочем потоке.

Без декоратора @pyqtSlot Qt замораживает "местоположение" рабочего экземпляра в тот момент, когда было установлено соединение: если оно было до moveToThread, оно привязывается к основному потоку, и код слота продолжает выполняться в основном потоке, даже если сигнал испускается после вызова moveToThread.

Соединения, сделанные после moveToThread связывают слот для выполнения рабочего потока в обоих случаях.

Код:

from PyQt5.QtCore import (QCoreApplication, QObject, QRunnable, QThread,
                          QThreadPool, pyqtSignal, pyqtSlot)

class Worker(QObject):
    def __init__(self):
        super(Worker, self).__init__()
#        self.call_f1.connect(self.f1)
#        self.call_f2.connect(self.f2)

    call_f1 = pyqtSignal()
    call_f2 = pyqtSignal()

    @pyqtSlot()
    def f1(self):
        print('f1', threading.get_ident())

    @pyqtSlot()
    def f2(self):
        print('f2', threading.get_ident())

app = QCoreApplication([])
print('main', threading.get_ident())
my_thread = QThread()
my_thread.start()

my_worker = Worker()
my_worker.call_f1.connect(my_worker.f1)
my_worker.call_f1.emit()
my_worker.moveToThread(my_thread)
my_worker.call_f2.connect(my_worker.f2)
my_worker.call_f1.emit()
my_worker.call_f2.emit()
sys.exit(app.exec_())

С декоратором:

main 18708
f1 18708
f1 20156
f2 20156

Без декоратора:

main 5520
f1 5520
f1 5520
f2 11472

PS Соединение в рабочем методе __init__ очевидно, эквивалентно соединению перед moveToThread в основном потоке.

(протестировано под PyQt5, win64).