通过信号的函数调用将默认键控参数更改为“ False”。为什么?

时间:2019-06-03 06:42:46

标签: python pyqt pyqt5 qpushbutton

当通过信号连接调用函数时,如

mybutton.clicked.connect(myfunction)

该函数的所有参数都设置为False。即使已设置默认参数。这是预期的行为吗?

下面的代码显示了一个简单的示例。对于我的特殊情况,将所有参数设置为False并不是一个大问题,因为我可以简单地使用if语句来捕获它。但是,我认为,了解为何以这种方式实施该方法可能对进一步的项目有所帮助。

这个简单的程序显示了我期望的行为。

def function(foo=None):
    print(foo)

function()
  

没有

然而,在这个最小的Qt示例中,functions参数不再是'None',而是'False'。

import sys
from PyQt5.QtWidgets import QMainWindow, QGridLayout, QWidget, QPushButton, QApplication

class MainWindow(QMainWindow):
    def __init__(self):
        QMainWindow.__init__(self)
        centralWidget = QWidget(self)
        self.setCentralWidget(centralWidget)
        gridLayout = QGridLayout(self)
        centralWidget.setLayout(gridLayout)
        btn = QPushButton("button",self)
        gridLayout.addWidget(btn)
        btn.clicked.connect(self.function)

    def function(self, foo=None):
        print(foo)

if __name__ == "__main__":
    app = QApplication(sys.argv)
    mainWin = MainWindow()
    mainWin.show()
    sys.exit( app.exec_() )
  

错误

我想的问题是:我是否必须忍受函数中没有可用的默认值(即,一旦我注意到它们设置为False,是否需要重置它们)还是有更聪明的方法?处理吗?

2 个答案:

答案 0 :(得分:3)

单击的是一个过载信号,即它们是两个具有相同名称和不同签名的信号。

clicked = QtCore.pyqtSignal([], [bool])
# clicked = QtCore.pyqtSignal()
# clicked = QtCore.pyqtSignal(bool)

有关更多详细信息,请阅读this answer

建立连接后,PyQt确定将使用哪个信号,在这种情况下,函数的参数类似于第二种形式,因此该信号将发送布尔False,在您的情况下它将替换默认值

如果您不希望发生这种情况,则必须使用@QtCore.pyqtSlot()装饰器强制PyQt使用第一种形式,以便它不会向您发送任何参数,因此您的函数将使用默认参数。

import sys
from PyQt5 import QtCore, QtWidgets


class MainWindow(QtWidgets.QMainWindow):
    def __init__(self, parent=None):
        super(MainWindow, self).__init__(parent)
        centralWidget = QtWidgets.QWidget()
        self.setCentralWidget(centralWidget)
        gridLayout = QtWidgets.QGridLayout(centralWidget)
        btn = QtWidgets.QPushButton("button")
        gridLayout.addWidget(btn)
        btn.clicked.connect(self.function)

    @QtCore.pyqtSlot()
    def function(self, foo=None):
        print(foo)


if __name__ == "__main__":
    app = QtWidgets.QApplication(sys.argv)
    mainWin = MainWindow()
    mainWin.show()
    sys.exit(app.exec_())

输出:

None

答案 1 :(得分:0)

另一种可能的解决方案是使用 * 语法强制 keyword-only parameters

class MainWindow(QtWidgets.QMainWindow):
    ...
    
    def function(self, *, foo=None):
        print(foo)