【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)
<function a at 0x0000018E05E276A8>

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方式用起来比较顺手。

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 简介 Qt :跨平台UI工具,用于开源软件,可以用自由软件协议GPL PyQt :是Qt为python提供的GUI...
    FreeRain77阅读 1,121评论 0 1
  • PyQt5入门教程 2019/12/11更新:我平时不看CSDN的,之前一时兴起发了过来,没想到反响还不错。这次就...
    资源分享吧1阅读 1,514评论 0 1
  • 指定一个处理程序后,就是将一个函数或方法的引用传递给connect()方法。如果需要传递数据给处理程序,可以按以下...
    用电热毯烤猪阅读 3,127评论 0 50
  • 信号和槽(Signals and Slots) Qt库第一个认识到在几乎所有情况下,程序员不需要或甚至不想知道所有...
    珞珈村下山阅读 9,893评论 0 23
  • 久违的晴天,家长会。 家长大会开好到教室时,离放学已经没多少时间了。班主任说已经安排了三个家长分享经验。 放学铃声...
    飘雪儿5阅读 7,574评论 16 22