QMutexLocker() 导致 UI 冻结

问题描述 投票:0回答:1

我有一个控制机器人的控制器类(通过串行接口连接)。该控制器附加到视图。除此之外,我还有一个源自

QThread
的线程,它会定期读取机器人的状态。

读取状态不得与从用户界面触发的机器人命令相冲突。因此,我使用

QMutexLocker
使用互斥体锁定了每个机器人访问,但这会导致我的用户界面在执行此类互斥体块时冻结。

class RobotControl(QObject):
    def __init__(self, view):
        super(RobotControl, self).__init__()
        self.view = view
        self.updatethread = UpdatePositionAndStatus(self.robot)
        self.mutex = QMutex()
        self.connect(self.updatethread, SIGNAL("updateStatus( QString ) "), self.print_error)
        self.updatethread.start()

@pyqtSlot()  
def init_robot(self):
    """
    Initializes the serial interface to the robot interface and checks if
    there is really a robot interface available.
    """
    with QMutexLocker(self.mutex):
        # Open interface
        try:
            index = self.view.robotcontrolui.get_selected_interface_index()
            interface = self.interfaces.item(index).text()
            self.robot = RobotController(interface)
        except DeviceError:
            self.view.error_dlg(self.tr("Couldn't open interface {0}!".format(interface)))
            self.robot = None
            return

        # Check if there is really a robot interface on the selected serial
        # interface with trying to read status byte
        try:
            self.robot.status()
        except DeviceError:
            # In case of failure release interface
            self.close_robot()
            self.view.error_dlg(self.tr("Couldn't initialize robot interface!"))
            return

        self.view.robotcontrolui.bt_open_interface.setEnabled(False)
        self.view.robotcontrolui.bt_close_interface.setEnabled(True)

class UpdatePositionAndStatus(QThread):
    def __init__(self, robot, parent=None):
        QThread.__init__(self, parent) 
        self.robot = robot
        self.mutex = QMutex()
    def run(self):
        """ 
        This function continously reads out the position and the status to for 
        updating it on the userinterface.
        """
        try:
            while True:
                if self.robot is not None:
                    # Do robot communication under a lock
                    self.mutex.lock()
                    (_, rel_pos) = self.robot.read_position()
                    status = self.robot.status()
                    self.mutex.unlock()

                    # Display position and status on userinterface
                    self.view.robotcontrolui.update_position_and_status(rel_pos, status)

                # Wait 1 seccond for next update
                QThread.sleep(1.0)
        except DeviceError:
            # Release lock on robot
            self.mutex.unlock()
            self.emit(SIGNAL("updateStatus( QString )"), self.tr("Error while updating current position and status!"))

触发 init 方法后,用户界面冻结并且程序崩溃:为什么会这样?我怎样才能避免这种情况?

python multithreading pyqt mutex robotics
1个回答
3
投票

很难说,因为您的代码示例不完整,但我发现此代码存在两个基本问题:

  1. 您正在锁定两个不同的 QMutex 对象。为了使互斥正常工作,两个线程必须锁定相同互斥对象。

  2. 您似乎在这一行的更新线程中直接与 GUI 交互:

    self.view.robotcontrolui.update_position_and_status(rel_pos, status)
    

    执行 GUI 操作只能从 Qt 中的 GUI 线程完成。可以肯定这是导致您崩溃的原因。请参阅:http://qt-project.org/doc/qt-4.8/threads.html

© www.soinside.com 2019 - 2024. All rights reserved.