在 PyQt5 的事件循环中使用 pyudev.pyqt5

Using pyudev.pyqt5 within PyQt5's event loop

我想写一个小应用程序,当它被插入时,它会在一个小的文本浏览器中显示 usb 设备的名称。我正在使用 pyudev 来做到这一点。我发现我们可以通过使用 MonitorObserver 将 pyudev 集成到 qt 的事件循环中,而不是使用 pyudev 自己的事件循环。我一直在努力这样做,但结果是徒劳的。 这是我到目前为止尝试过的代码。如果我犯了错误,请告诉我。

进口

from PyQt5.QtCore import pyqtSignal,pyqtSlot,QObject,QSocketNotifier
from PyQt5.QtWidgets import QWidget,QApplication
from form_designer import Ui_Form
from pyudev.pyqt5 import MonitorObserver
from pyudev import Context,Monitor
from PyQt5 import QtCore

小Class

class mainWindow(QWidget,Ui_Form):
    def __init__(self,monitor):
        super().__init__()

        self.setupUi(self)
        print("First test")

        monitor.filter_by(subsystem='tty')
        self.observer = MonitorObserver(monitor)
        print("second test")

        self.observer.deviceEvent.connect(self.device_connected)
    @pyqtSlot()
    def device_connected(self,device):
        self.textBrowser.append(device.sys_name)
        print("Test")

我什至使用打印进行测试以查看循环进行了多远。 这里是初始化和main

def main():
    import sys
    app = QApplication(sys.argv)
    context = Context()
    monitor = Monitor.from_netlink(context)
    window = mainWindow(monitor)
    window.show()

    app.exec_()
    monitor.start()
if __name__ == '__main__':
    main()

执行时,它会打印前两个打印件并且永远不会调用插槽。

仅供参考:自从我开始学习以来已经一周了 python。

您需要在启动事件循环之前启动监视器。另外,避免使用 @pyqtSlot 装饰器 - 它很少需要,如果您不知道自己在做什么(就像您在示例中所做的那样),很容易弄错定义。

因此您的代码应如下所示:

class mainWindow(QWidget, Ui_Form):
    def __init__(self):
        ...
        context = Context()
        monitor = Monitor.from_netlink(context)
        monitor.filter_by(subsystem='tty')
        self.observer = MonitorObserver(monitor)
        self.observer.deviceEvent.connect(self.device_connected)
        monitor.start()

    def device_connected(self, device):
        self.textBrowser.append(device.sys_name)
        print("Test")

def main():
    import sys
    app = QApplication(sys.argv)
    window = mainWindow()
    window.show()
    app.exec_()

PS:最好注释掉 filter_by 行并插入 USB 设备以确保一切正常。