------------------
-
问题
最近在学习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))
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方式用起来比较顺手。