当前位置: 首页 > 面试题库 >

Lambda中连接到pyqtSignal的对象的生命周期

邹嘉荣
2023-03-14
问题内容

假设我有一个对象,并希望在发出PyQt信号时执行其方法之一。并假设我希望它使用信号未传递的参数来执行此操作。因此,我创建了一个lambda作为信号的插槽:

class MyClass(object):
    def __init__(self, model):
        model.model_changed_signal.connect(lambda: self.set_x(model.x(), silent=True))

现在,通常使用PyQt信号和插槽,信号连接不会阻止垃圾收集。当连接的插槽的对象被垃圾回收时,发出相应信号时将不再调用该插槽。

但是,使用lambdas时如何工作?我没有存储对lambda的引用,但是信号插槽连接确实可以正常工作。因此,lambda不会被垃圾收集。

如果我现在将的实例设置MyClassNone,则该实例也不会被垃圾收集:发出model_changed_signal静止图像会成功执行lambda。因此,显然,对Instance的引用MyClass保留在某个位置(也许在lambda的上下文中)周围-
我不希望这样。

为什么会这样?


问题答案:

lambda您的示例中形成一个闭合。也就是说,它是一个嵌套函数,引用了其封闭范围内的可用对象。每个创建闭包的函数都会为其需要维护的每个项目保留一个单元格对象。

在您的示例中,lambda会创建一个闭包,并引用方法范围内的局部变量selfmodel变量__init__。如果保留对lambda某处的引用,则可以通过其__closure__属性检查其关闭的所有单元格对象。在您的示例中,它将显示如下内容:

>>> print(func.__closure__)
(<cell at 0x7f99c16c5138: MyModel object at 0x7f99bbbf0948>, <cell at 0x7f99c16c5168: MyClass object at 0x7f99bbb81390>)

如果删除了此处显示的MyModelMyClass对象的所有其他引用,则单元格保留的那些引用仍将保留。因此,当涉及对象清除时,您应始终明确断开与可能会在相关对象上形成闭合的函数连接的所有信号。

请注意,在信号/插槽连接方面,PyQt对包装的C ++插槽和Python实例方法的处理方式有所不同。当这些可调用类型连接到信号时,它们的引用计数
不会
增加,而lambda,已定义函数,部分对象和静态方法会增加。这意味着,如果删除了对后一种可调用类型的所有其他引用,则所有剩余的信号连接将使它们保持活动状态。断开信号将允许在必要时对这些已连接的可调用对象进行垃圾回收。

上面的一个例外是类方法。PyQt在创建与它们的连接时会创建一个特殊的包装器,因此,如果删除了对它们的所有其他引用,并且发出了信号,则会引发异常,如下所示:

TypeError: 'managedbuffer' object is not callable

以上内容应适用于PyQt5和大多数版本的PyQt4(4.3及更高版本)。



 类似资料:
  • 每个响应对象只有当在 servlet 的 service 方法的范围内或在 filter 的 doFilter 方法范围内是有效的,除非该组件关联的请求对象已经开启异步处理。如果相关的请求已经启动异步处理,那么直到AsyncContext 的 complete 方法被调用,请求对象一直有效。为了避免响应对象创建的性能开销,容器通常回收响应对象。在相关的请求的startAsync 还没有调用时,开发

  • 每个请求对象只在一个 servlet 的 service 方法的作用域内,或过滤器的 doFilter 方法的作用域内有效,除非该组件启用了异步处理并且调用了请求对象的 startAsync 方法。在发生异步处理的情况下,请求对象一直有效,直到调用 AsyncContext 的 complete 方法。容器通常会重复利用请求对象,以避免创建请求对象而产生的性能开销。开发人员必须注意的是,不建议在上

  • Hikari:2.4.7 PostgreSQL JDBC驱动程序:9.4-1201-jdbc41 我试图理解必须对java.sql做些什么。对象以使其在连接池中再次可用? 我刚刚将连接池引入了一个多线程应用程序,该应用程序以前使用每个SQL语句建立/拆除连接。 在介绍Hikari之后,我注意到,一旦我点击,此后每次尝试都会由于而失败。所以看起来我没有以某种方式“释放”这个连接。 连接对象的典型用法

  • 一旦一个类被装载、连接和初始化,它就随时可以被使用。程序可以访问它的静态字段,调用它的静态方法,或者创建它的实例。作为Java程序员有必要了解Java对象的生命周期。 类实例化 在Java程序中,类可以被明确或隐含地实例化。明确的实例化类有四种途径: 明确调用new。 调用Class或者java.lang.reflect.Constructor对象的newInstance方法。 调用任何现有对象的

  • C 没有GC机制,当C对象被导出到JavaScript环境后,必须使用某种方法进行对象生命周期管理,以彻底杜绝野指针、内存泄漏,引用计数无疑是最常用的方法。 4.4.1 引用计数基类 对象生命周期管理需要解决的问题是:当一个对象可能在多个地方被引用时,如何决定何时将其销毁。引用计数法解决这一问题的途径非常简单: 每个对象自带一个初值为0的引用计数; 对象的每个使用者,在获得一个对象的引用时,将其引

  • 我需要从当前feature文件调用一个.feature文件,并将一个变量从被调用的.feature文件传递给调用方.feature文件。我使用了karate.set()和karate.get(),但变量似乎不是这样传递的。在调用的.feature文件中,我将变量设置为“*def token=karate.get('xenpauth')”。在调用的.feature文件中,我尝试获得类似“*defxe