PySide / PyQt - запуск потока с интенсивным использованием процессора приводит к зависанию всего приложения

Я пытаюсь сделать довольно распространенную вещь в моем приложении с графическим интерфейсом PySide: я хочу делегировать некоторую интенсивную ЦП задачу фоновому потоку, чтобы мой графический интерфейс оставался отзывчивым и мог даже отображать индикатор прогресса в процессе вычислений.

Вот что я делаю (я использую PySide 1.1.1 на Python 2.7, Linux x86_64):

import sys
import time
from PySide.QtGui import QMainWindow, QPushButton, QApplication, QWidget
from PySide.QtCore import QThread, QObject, Signal, Slot

class Worker(QObject):
    done_signal = Signal()

    def __init__(self, parent = None):
        QObject.__init__(self, parent)

    @Slot()
    def do_stuff(self):
        print "[thread %x] computation started" % self.thread().currentThreadId()
        for i in range(30):
            # time.sleep(0.2)
            x = 1000000
            y = 100**x
        print "[thread %x] computation ended" % self.thread().currentThreadId()
        self.done_signal.emit()


class Example(QWidget):

    def __init__(self):
        super(Example, self).__init__()

        self.initUI()

        self.work_thread = QThread()
        self.worker = Worker()
        self.worker.moveToThread(self.work_thread)
        self.work_thread.started.connect(self.worker.do_stuff)
        self.worker.done_signal.connect(self.work_done)

    def initUI(self):

        self.btn = QPushButton('Do stuff', self)
        self.btn.resize(self.btn.sizeHint())
        self.btn.move(50, 50)       
        self.btn.clicked.connect(self.execute_thread)

        self.setGeometry(300, 300, 250, 150)
        self.setWindowTitle('Test')    
        self.show()


    def execute_thread(self):
        self.btn.setEnabled(False)
        self.btn.setText('Waiting...')
        self.work_thread.start()
        print "[main %x] started" % (self.thread().currentThreadId())

    def work_done(self):
        self.btn.setText('Do stuff')
        self.btn.setEnabled(True)
        self.work_thread.exit()
        print "[main %x] ended" % (self.thread().currentThreadId())


def main():

    app = QApplication(sys.argv)
    ex = Example()
    sys.exit(app.exec_())


if __name__ == '__main__':
    main()

Приложение отображает одно окно с кнопкой. Когда кнопка нажата, я ожидаю, что она отключится сама, пока выполняется вычисление. Затем кнопка должна быть снова включена.

Вместо этого происходит то, что, когда я нажимаю кнопку, все окно останавливается во время вычислений, а затем, когда оно завершается, я возвращаю управление приложению. Кнопка никогда не кажется отключенной. Забавно, что я заметил, что, если я заменю процессор интенсивным вычислением вdo_stuff() с простым time.sleep () программа ведет себя как ожидалось.

Я точно не знаю, что происходит, но кажется, что приоритет второго потока настолько высок, что фактически предотвращает планирование потока GUI. Если второй поток переходит в состояние BLOCKED (как это происходит сsleep()), графический интерфейс на самом деле имеет шанс запустить и обновить интерфейс, как и ожидалось. Я попытался изменить приоритет рабочего потока, но похоже, что это невозможно сделать в Linux.

Кроме того, я пытаюсь напечатать идентификаторы потоков, но я не уверен, правильно ли я это делаю. Если да, то сродство нитей кажется правильным.

Я также попробовал программу с PyQt, и поведение точно такое же, отсюда и теги, и заголовок. Если бы я мог заставить его работать с PyQt4 вместо PySide, я мог бы переключить все свое приложение на PyQt4

Ответы на вопрос(2)

Ваш ответ на вопрос