尽管就Python用户而言,Python信号处理程序是异步调用的,但它们只能出现在Python解释器的“原子”指令之间。这意味着纯粹在C语言中执行的长计算期间到达的信号(例如,大文本正文中的正则表达式匹配)可能会延迟任意时间。
这意味着在Qt事件循环运行时,Python无法处理信号。仅当Python解释器运行时(退出QApplication或从Qt调用Python函数时),才会调用信号处理程序。
一种解决方案是使用QTimer让解释器不时运行。
请注意,在下面的代码中,如果没有打开的窗口,则应用程序将在消息框后退出,而不考虑用户的选择,因为QApplication.quitOnLastWindowClosed()== True。可以更改此行为。
import signal
import sys
from PyQt4.QtCore import QTimer
from PyQt4.QtGui import QApplication, QMessage@R_221_2419@
# Your code here
def sigint_handler(*args):
"""Handler for the SIGINT signal."""
sys.stderr.write('\r')
if QMessage@R_221_2419@.question(None, '', "Are you sure you want to quit?",
QMessage@R_221_2419@.Yes | QMessage@R_221_2419@.No,
QMessage@R_221_2419@.No) == QMessage@R_221_2419@.Yes:
QApplication.quit()
if __name__ == "__main__":
signal.signal(signal.SIGINT, sigint_handler)
app = QApplication(sys.argv)
timer = QTimer()
timer.start(500) # You may change this if you wish.
timer.timeout.connect(lambda: None) # Let the interpreter run each 500 ms.
# Your code here.
sys.exit(app.exec_())
如LinearOrbit所指出的,另一个可能的解决方案是signal.signal(signal.SIGINT, signal.SIG_DFL)
,但是它不允许自定义处理程序。
如果您只是希望ctrl-c关闭应用程序-而不是“很好” /优雅-然后从http://www.mail- archive.com/pyqt@riverbankcomputing.com/msg13758.html,可以使用这个:
import signal
signal.signal(signal.SIGINT, signal.SIG_DFL)
import sys
from PyQt4.QtCore import QCoreApplication
app = QCoreApplication(sys.argv)
app.exec_()
显然,这可以在Linux,Windows和OSX上使用-到目前为止,我仅在Linux上对其进行过测试(并且可以使用)。