3

我想写一个小应用程序,当它被插入时,它会在一个小文本浏览器中显示 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 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")

我什至使用 prints 进行了测试,以查看循环进行了多远。这是初始化和主要

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 已经一周了。

4

1 回答 1

4

您需要在启动事件循环之前启动监视器。此外,避免使用@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设备以确保一切正常可能是个好主意。

于 2016-11-26T02:00:53.233 回答