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

use*_*816 5 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_())
Run Code Online (Sandbox Code Playgroud)

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

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

  • 如何按照陷阱页面的建议使对象成为我的窗口或小部件的属性?
  • 虽然使其成为窗口/小部件的属性是一种选择,但是否有更好的设计/更优雅的解决方案/最佳实践可供使用?

谢谢您的帮助。

ekh*_*oro 5

创建一个子类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_())
Run Code Online (Sandbox Code Playgroud)


nto*_*pos 5

它正在被垃圾收集,因为一旦函数执行结束,作用域就会丢失。您需要保存引用,有两种方法可以做到:

  1. 将对象保存在一个全局变量中,这样它就不会被 gc 收集,并且在其他函数上也保持可访问性。
  2. 将整个事情包装在一个QWidget实现中,将创建的对象保存在小部件对象中(通过 self)。

如果我不够清楚或不够准确,请告诉我。
希望能帮助到你。
干杯。