ICode9

精准搜索请尝试: 精确搜索
首页 > 编程语言> 文章详细

python-pyqt:将多个信号连接到pyqt中的同一函数的正确方法(QSignalMapper不适用)

2019-11-19 14:57:07  阅读:454  来源: 互联网

标签:pyqt signals-slots python garbage-collection qsignalmapper


>我已经准备好许多关于如何将多个信号连接到python和pyqt中的同一事件处理程序的文章.例如,将多个按钮或组合框连接到同一功能.
>许多示例显示了如何使用QSignalMapper进行此操作,但是不适用于信号带有参数的情况,例如combobox.currentIndexChanged
>许多人建议可以用lambda制成.我同意,这是一个干净漂亮的解决方案,但是没有人提及lambda创建一个闭包,该闭包包含一个引用-因此无法删除被引用的对象.你好,内存泄漏!

证明:

from PyQt4 import QtGui, QtCore

class Widget(QtGui.QWidget):
    def __init__(self):
        super(Widget, self).__init__()

        # create and set the layout
        lay_main = QtGui.QHBoxLayout()
        self.setLayout(lay_main)

        # create two comboboxes and connect them to a single handler with lambda

        combobox = QtGui.QComboBox()
        combobox.addItems('Nol Adyn Dwa Tri'.split())
        combobox.currentIndexChanged.connect(lambda ind: self.on_selected('1', ind))
        lay_main.addWidget(combobox)

        combobox = QtGui.QComboBox()
        combobox.addItems('Nol Adyn Dwa Tri'.split())
        combobox.currentIndexChanged.connect(lambda ind: self.on_selected('2', ind))
        lay_main.addWidget(combobox)

    # let the handler show which combobox was selected with which value
    def on_selected(self, cb, index):
        print '! combobox ', cb, ' index ', index

    def __del__(self):
        print 'deleted'

if __name__ == '__main__':

    import sys
    app = QtGui.QApplication(sys.argv)

    wdg = Widget()
    wdg.show()

    wdg = None

    sys.exit(app.exec_())

尽管我们清除了引用,但未删除该小部件.删除与lambda的连接-它将正确删除.

因此,问题是:在不泄漏内存的情况下,将具有参数的多个信号连接到单个处理程序的正确方法是什么?

解决方法:

不能删除对象是不正确的,因为信号连接在闭包中保留了引用. Qt在删除对象时会自动删除所有信号连接,这反过来又会删除python端对lambda的引用.

但这意味着您不能总是仅依靠Python来删除对象.每个PyQt对象都有两部分:Qt C部分和Python包装器部分.这两个部分都必须删除-有时以特定顺序删除(取决于Qt或Python当前是否拥有该对象的所有权).除此之外,还有Python垃圾收集器的各种变化因素(尤其是在解释器关闭的短时间内).

无论如何,在您的特定示例中,简单的解决方法是:

    # wdg = None
    wdg.deleteLater()

这计划删除对象,因此需要运行事件循环才能生效.在您的示例中,这还将自动退出应用程序(因为该对象是最后关闭的窗口).

为了更清楚地了解正在发生的事情,您还可以尝试以下操作:

    #wdg = None
    wdg.deleteLater()

    app.exec_()

    # Python part is still alive here...
    print(wdg)
    # but the Qt part has already gone
    print(wdg.objectName())

输出:

<__main__.Widget object at 0x7fa953688510>
Traceback (most recent call last):
  File "test.py", line 45, in <module>
    print(wdg.objectName())
RuntimeError: wrapped C/C++ object of type Widget has been deleted
deleted

编辑:

这是另一个调试示例,希望可以使其更加清晰:

    wdg = Widget()
    wdg.show()

    wdg.deleteLater()
    print 'wdg.deleteLater called'

    del wdg
    print 'del widget executed'

    wd2 = Widget()
    wd2.show()

    print 'starting event-loop'
    app.exec_()

输出:

$python2 test.py
wdg.deleteLater called
del widget executed
starting event-loop
deleted

标签:pyqt,signals-slots,python,garbage-collection,qsignalmapper
来源: https://codeday.me/bug/20191119/2036949.html

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有