连接到 pyqtSignal 的 lambda 中的对象的生命周期
- 2025-04-15 09:19:00
- admin 原创
- 30
问题描述:
假设我有一个对象,并希望在 PyQt 信号发出时执行它的一个方法。并且假设我希望它使用一个信号未传递的参数来执行此操作。因此,我创建一个 lambda 作为信号的槽函数:
class MyClass(object):
def __init__(self, model):
model.model_changed_signal.connect(lambda: self.set_x(model.x(), silent=True))
现在,在 PyQt 信号和槽中,信号连接通常不会阻止垃圾回收。当连接的槽对象被垃圾回收时,在发出相应信号时,该槽将不再被调用。
但是,使用 lambda 表达式时该如何实现呢?我没有存储 lambda 表达式的引用,但信号槽连接仍然有效。所以 lambda 表达式不会被垃圾回收。
如果我现在将 的实例设置MyClass
为None
,该实例也不会被垃圾回收:发出model_changed_signal
仍然会成功执行 lambda。因此,显然,对 实例的引用MyClass
会保留在某个地方(也许是在 lambda 的上下文中?)——这是我不想要的。
为什么会发生这种情况?
解决方案 1:
示例lambda
中的 构成了一个闭包。也就是说,它是一个嵌套函数,引用了其封闭作用域内可用的对象。每个创建闭包的函数都会为其需要维护引用的每个项保留一个cell 对象。
在您的示例中, 会创建一个闭包,其中包含对方法范围内的lambda
局部变量self
和变量的引用。如果您保留对 某个地方 的引用,则可以通过 的属性检查其闭包中的所有单元格对象。在您的示例中,它将显示如下内容:model
`__init__lambda
__closure__`
>>> print(func.__closure__)
(<cell at 0x7f99c16c5138: MyModel object at 0x7f99bbbf0948>, <cell at 0x7f99c16c5168: MyClass object at 0x7f99bbb81390>)
如果您删除了此处所示的所有其他对MyModel
和MyClass
对象的引用,单元格保留的引用仍将保留。因此,在进行对象清理时,应始终明确断开所有连接到可能在相关对象上形成闭包的函数的信号。
请注意,在信号/槽连接方面,PyQt 对包装的 C++ 槽和 Python 实例方法的处理方式有所不同。这些类型的可调用函数在连接到信号时不会增加引用计数,而 lambda 表达式、已定义函数、部分对象和静态方法则会。这意味着,如果删除了对后者可调用函数的所有其他引用,任何剩余的信号连接都将使它们保持活动状态。断开信号连接将允许此类连接的可调用函数在必要时被垃圾回收。
上述方法的一个例外是类方法。PyQt 在创建与这些方法的连接时会创建一个特殊的包装器,因此,如果所有对它们的引用都被删除,并且发出了信号,则会引发异常,如下所示:
TypeError: 'managedbuffer' object is not callable
以上内容适用于 PyQt5 和大多数版本的 PyQt4(4.3 及更高版本)。
扫码咨询,免费领取项目管理大礼包!