Поток Pyqt5 не может работать (фатальная ошибка Python: не удалось получить блокировку) - PullRequest
0 голосов
/ 19 декабря 2018

Я прочитал кучу руководств по QThread и SO Ответов по этому поводу.Но я не могу заставить мои потоки работать.В большинстве случаев он запускается один раз или выдает сообщение об ошибке.Потоки работают, если я запустил их сразу после запуска приложения, но я хочу, чтобы поток запускался после определенной функции, потому что я хочу сначала указать местоположение каталога.

Структура файла устроена так:

App.py
Controllers/
    main_controller.py
    recorder.py
Model/
    model.py
Views/
    main_view.py

App.py

import sys
from PyQt5.QtWidgets import QApplication

from Model.model import Model
from Controllers.main_controller import MainController
from Views.main_view import MainView


class App(QApplication):
    def __init__(self, sys_argv):
        super().__init__(sys_argv)
        self.model = Model()
        self.main_controller = MainController(self.model)
        self.main_view = MainView(self.model, self.main_controller)


if __name__ == "__main__":
    app = App(sys.argv)
    sys.exit(app.exec_())

model.py

from PyQt5.QtCore import QObject


class Model(QObject):
    def __init__(self):
        super().__init__()
        self.directory = ""

    def get_directory(self):
        return self.directory

    def set_directory(self, directory):
        self.directory = directory

main_view.py

from PyQt5.QtWidgets import QMenu, QSystemTrayIcon, QMainWindow, QFileDialog
from PyQt5 import QtGui
from PyQt5.QtWidgets import QApplication
from Controllers.recorder import Recorder


class MainView(QMainWindow):

    def __init__(self, model, main_controller):
        super().__init__()

        self._model = model
        self._main_controller = main_controller

        # UI
        icon = QtGui.QIcon("icon24x24.png")
        menu = QMenu()
        start_action = menu.addAction("Start Recording")
        stop_action = menu.addAction("Stop Recording")
        self.tray = QSystemTrayIcon()
        self.tray.setIcon(icon)
        self.tray.setContextMenu(menu)
        self.tray.show()

        start_action.triggered.connect(self.start_app)
        stop_action.triggered.connect(self.stop_app)

        self.recordThread = Recorder()

    def start_app(self):
        directory = QFileDialog.getExistingDirectory(self, "Select Directory")
        self._main_controller.set_directory(directory)
        self.start_thread()

    def start_thread(self):
        self.recordThread.start()

    def stop_app(self):
        self.recordThread.terminate()
        QApplication.instance().quit()
        print("app stopped")

main_controller.py

from PyQt5.QtCore import QObject


class MainController(QObject):

    def __init__(self, model):
        super().__init__()
        self._model = model

    def set_directory(self, directory):
        self._model.set_directory(directory)

Recorder.py

import time
from PyQt5.QtCore import QThread, QTimer, pyqtSignal
from Model.model import Model


class Recorder(QThread):

    job_done = pyqtSignal()

    def __init__(self):
        QThread.__init__(self)
        self._model = Model()

    def __del__(self):
        self.wait()

    def run(self):
        while True:
            print("I am the loop")
            print(self._model.get_directory())
            # time.sleep(4 - time.time() % 4)
            QThread.sleep(4)
            print("now is {}".format(time.time()))
        self.job_done.emit()

Я пробовалс использованием различных стилей, включая Qthread, QObject, pyqtsignal в соответствии с различными учебными пособиями.но у меня ничего не работает.Либо просто выведите «Я петля», затем выйдите.или распечатайте

I am the loop
Fatal Python error: could not acquire lock for <_io.BufferedWriter name='<stdout>'> at interpreter shutdown, possibly due to daemon threads

Thread 0x00007f5d1bfff700 (most recent call first):
  File "/App/Controllers/recorder.py", line 20 in run

Current thread 0x00007f5d3458f700 (most recent call first):
Aborted

Спасибо

1 Ответ

0 голосов
/ 20 декабря 2018

Ничего плохого в вашем коде.Приложение закрылось сразу после вызова QFileDialog.getExistingDirectory, потому что это природа Qt.

Qt сделан для выхода, когда все окна закрыты

Приложение закрыто, потому что у вас больше нет окон.В вашем приложении нет окон, но QSystemTrayIcon().Установка setQuitOnLastWindowClosed() в False решила проблему.

yourApp.setQuitOnLastWindowClosed(False)

Источник

...