对象超出范围并在PySide / PyQt中收集垃圾

时间:2016-08-12 18:02:12

标签: python qt garbage-collection pyqt pyside

我遇到的问题是非Qt对象超出范围并被Python垃圾收集。在下面的通用示例中,我在单击一个按钮时创建一个对象,并在单击另一个按钮时想要使用该对象。在第一个按钮的插槽完成执行后,该对象被垃圾收集。

# Started from http://zetcode.com/gui/pysidetutorial/firstprograms/ and
# connections program from chap 4 of Summerfield's PyQt book

import sys
from PySide import QtGui


def createThingy():
    thing1 = thingy("thingName", 3, wid)
    lbl1.setText(thing1.name + " created.")
    return


def updateNum():
    lbl1.setText("number: " + str(thing1.number))

    return


class thingy(object):
    def __init__(self, name, number, mainWindow):
        self.name = name
        self.number = number

    def func1(self):
        return "Something to return"

    def rtnNum(self):
        return "Num: " + str(self.number)

app = QtGui.QApplication(sys.argv)

wid = QtGui.QWidget()
wid.resize(250, 150)
wid.setWindowTitle('Example')

btn1 = QtGui.QPushButton("Create thingy")
btn2 = QtGui.QPushButton("Number")
lbl1 = QtGui.QLabel("---")

vlayout = QtGui.QVBoxLayout()
vlayout.addWidget(btn1)
vlayout.addWidget(btn2)
vlayout.addWidget(lbl1)
wid.setLayout(vlayout)

btn1.clicked.connect(createThingy)
btn2.clicked.connect(updateNum)

wid.show()
wid.raise_()
sys.exit(app.exec_())

在PySide陷阱页面https://wiki.qt.io/PySide_Pitfalls上提到了这种行为,以及“Python PySide(内部c ++对象已经删除)”https://stackoverflow.com/a/5339238/2599816这个问题(这个问题没有真正回答我的问题,因为作者的解决方案是避免这个问题。)

因为该对象不是Qt对象,所以将父对象传递给它不是一个可行的选择。另外,我不想在程序开始时创建对象,因为其他一些问题/答案已在其他地方建议(无法再找到它们的链接)。此外,不应修改对象以解决此问题,但应在GUI中实现解决方案,即保持对象通用以用作其他程序中的库。

  • 如何根据陷阱页面的建议将对象作为我的窗口或窗口小部件的属性?
  • 虽然将窗口/窗口小部件的属性作为一个选项,是否有更好的设计/更优雅的解决方案/最佳实践可供使用?

感谢您的帮助。

2 个答案:

答案 0 :(得分:4)

创建QWidget的子类以将所有子对象保留为属性:

import sys
from PySide import QtGui

class Widget(QtGui.QWidget):
    def __init__(self, parent=None):
        super(Widget, self).__init__(parent)
        self.resize(250, 150)
        self.setWindowTitle('Example')

        self.btn1 = QtGui.QPushButton("Create thingy")
        self.btn2 = QtGui.QPushButton("Number")
        self.lbl1 = QtGui.QLabel("---")

        vlayout = QtGui.QVBoxLayout()
        vlayout.addWidget(self.btn1)
        vlayout.addWidget(self.btn2)
        vlayout.addWidget(self.lbl1)
        self.setLayout(vlayout)

        self.btn1.clicked.connect(self.createThingy)
        self.btn2.clicked.connect(self.updateNum)

    def createThingy(self):
        self.thing1 = thingy("thingName", 3, wid)
        self.lbl1.setText(self.thing1.name + " created.")

    def updateNum(self):
        self.lbl1.setText("number: " + str(self.thing1.number))

class thingy(object):
    def __init__(self, name, number, mainWindow):
        self.name = name
        self.number = number

    def func1(self):
        return "Something to return"

    def rtnNum(self):
        return "Num: " + str(self.number)

if __name__ == '__main__':

    app = QtGui.QApplication(sys.argv)
    wid = Widget()
    wid.show()
    sys.exit(app.exec_())

答案 1 :(得分:3)

它被垃圾收集,因为一旦函数执行结束,范围就会丢失。您需要保存引用,有两种方法:

  1. 将对象保存在全局变量中,因此它不会被gc收集,也可以在其他函数中保持可访问状态。
  2. 将整个事物包裹在QWidget实现中,将创建的对象保存在widget对象中(通过self)。
  3. 如果我不清楚或不够准确,请告诉我。
    希望能帮助到你。
    欢呼声。