【PyQt5】为槽函数传递额外参数的方法

------------------
  • 问题

最近在学习pyqt的时候遇到了一个问题,在类外编写某个信号的带参数的槽函数,同时在外部建立信号和槽的连接时,无法传递参数。
描述的不清楚,直接上代码

    def openfile(self, widget):
        path = QtWidgets.QFileDialog.getExistingDirectory(widget, "浏览", "./")
        print(path)

上面这段代码就是我定义的槽函数,调用打开文件获取文件夹路径的操作,该函数中需要传递一个Widget作为参数,但是在使用的时候,传递的是函数名,而不是函数返回值(函数运算结果)

b.pushButton.clicked.connect(slots.openfile)

很明显,这样调用函数是无法达到想要的效果的,没有传递参数的话,在界面上点击按钮后,会直接报错退出

Process finished with exit code -1073740791 (0xC0000409)

那我们加上参数调用会怎么样呢?

a = QtWidgets.QWidget()
b.pushButton.clicked.connect(slots.openfile(a))
直接出现选择窗口.png
    b.pushButton.clicked.connect(slots.openfile(a))
TypeError: argument 1 has unexpected type 'NoneType'

加上参数后,运行直接出现选择窗口,QT界面已经没有了,更没有信号槽连接了,点击选择或者取消后,报错信息为参数unexpected type 'NoneType'。

-------------------------

  • 如何传递这种带参数的SLOT呢

我们先看一下connect函数的定义

    def connect(self, slot, type=None, no_receiver_check=False): # real signature unknown; restored from __doc__
        """
        connect(slot, type=Qt.AutoConnection, no_receiver_check=False)
        
        slot is either a Python callable or another signal.
        type is a Qt.ConnectionType.
        no_receiver_check is True to disable the check that the receiver's C++
        instance still exists when the signal is emitted.
        """
        pass

slot is either a Python callable or another signal.

slot这个参数可以是Python的callable,或者是其他的信号。这个不难理解,slot这个参数是一个可以调用的python函数,来执行函数操作,或者是一个其他的信号,这样就可以使用信号1来触发信号2或者信号X。
而什么是callable函数呢?callable是Python的一个内置函数,其描述为:

对于函数、方法、lambda 函式、 类以及实现了 __call__ 方法的类实例, 它都返回 True。

结合connect的slot参数,我们来看一下callable()函数的返回值:

In [39]: def test(x):
    ...:     return x**2
    ...:

In [40]: callable(test)
Out[40]: True

In [41]: callable(test(2))
Out[41]: False

通过代码测试可以看出,test是callable对象,而test(2),带上参数的对象就不是callable的。同样 openfile(self, widget)函数中,openfile就是可调用的,而加上参数的openfile(a)就是不可调用的,如何将openfile(a)变成可调用的对象呢?
首先来看看test和test(2)的区别:

In [42]: print(test)


In [43]: print(test(2))
4

上面代码可以看出,test是一个函数,返回函数地址,而test(2)是返回值,返回计算结果。connect函数接收的是一个函数地址,而不是一个函数值。接下来只需要将openfile(a)变成一个函数地址就可以被connect接收了。

-------------------------

  • 传递带参数函数的两种方法

1.使用lambda匿名表达式
虽然lambda的争议很多,但我觉得用到回调函数或者槽函数这里刚好合适。

    b.pushButton.clicked.connect(lambda: slots.openfile(a))
In [44]: callable(lambda:test(2))
Out[44]: True

使用lambda将带参数的函数对象变为一个callable对象,而不是一个值,这种方法简单且可读性也不错。
2.使用partial偏函数方式

 b.pushButton.clicked.connect(partial(slots.openfile,a))
In [46]: callable(functools.partial(a,2))
Out[46]: True

python的函数增强器,functools中的partial函数可以实现同样的效果,但是需要引入functools包,同样很好用。

使用哪种方式还是看个人习惯,笔者觉得lambda方式用起来比较顺手。

你可能感兴趣的:(【PyQt5】为槽函数传递额外参数的方法)