捕获 QApplication 中引发的异常

Mik*_*ski 10 python exception pyqt pyqt5

我正在尝试使用 PyQt5 编写一个在系统托盘中工作的应用程序。代码有时会引发异常,我需要能够捕获它们。

我希望当应用程序中发生异常时,主事件循环会退出,因此像这样捕获它应该可以工作:

try:
    application.exec()
except:
    do_stuff()
Run Code Online (Sandbox Code Playgroud)

在下面的例子中,当我按下“Raise”按钮时,我只看到了回溯,但我从来没有看到error catched!打印出来的。

from PyQt5 import QtWidgets, QtGui, QtCore

class ErrorApp():
    def __init__(self):
        # Init QApplication, QWidet and QMenu
        self.app = QtWidgets.QApplication([])
        self.widget = QtWidgets.QWidget()
        self.menu = QtWidgets.QMenu("menu", self.widget)

        # Add items to menu
        self.menu_action_raise = self.menu.addAction("Raise")
        self.menu_action_raise.triggered.connect(self.raise_error)

        self.menu_action_exit = self.menu.addAction("Exit")
        self.menu_action_exit.triggered.connect(self.app.exit)

        # Create the tray app
        self.tray = QtWidgets.QSystemTrayIcon(QtGui.QIcon("logo.png"), self.widget)
        self.tray.setContextMenu(self.menu)

        # Show app
        self.tray.show()

    def raise_error(self):
        assert False

e = ErrorApp()

try:
    e.app.exec()

except:
    print("error catched!")
Run Code Online (Sandbox Code Playgroud)

有 2 个类似的问题,但那里的答案没有做我需要做的:

抓住 PyQt 中的任何异常:OP 想要监视异常,偶数循环没有退出 防止 PyQt 使插槽中发生的异常静音:装饰器答案根本不起作用;添加sys.exit(1)sys.excepthook只是关闭整个程序,不打印error catched!

eyl*_*esc 19

您必须使用异常,并且如果您希望事件循环结束,则必须调用quit()(or exit()) 方法。

import sys
import traceback
from PyQt5 import QtWidgets, QtGui, QtCore


class ErrorApp:
    # ...

    def raise_error(self):
        assert False


def excepthook(exc_type, exc_value, exc_tb):
    tb = "".join(traceback.format_exception(exc_type, exc_value, exc_tb))
    print("error catched!:")
    print("error message:\n", tb)
    QtWidgets.QApplication.quit()
    # or QtWidgets.QApplication.exit(0)


sys.excepthook = excepthook
e = ErrorApp()
ret = e.app.exec_()
print("event loop exited")
sys.exit(ret)
Run Code Online (Sandbox Code Playgroud)

输出:

error catched!:
error message:
 Traceback (most recent call last):
  File "main.py", line 28, in raise_error
    assert False
AssertionError

event loop exited
Run Code Online (Sandbox Code Playgroud)

  • @mikerodent`quit()`确实出现在QApplication成员列表中,quit()在primaryScreenChanged之后的第一列中,似乎顺序是首先放置属性,信号和槽,然后是其他方法。注意:槽是在 QMetaObject 中注册的函数。 (2认同)