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 中实现该解决方案,即保持对象通用以用作其他程序中的库。
谢谢您的帮助。
创建一个子类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)
它正在被垃圾收集,因为一旦函数执行结束,作用域就会丢失。您需要保存引用,有两种方法可以做到:
QWidget实现中,将创建的对象保存在小部件对象中(通过 self)。如果我不够清楚或不够准确,请告诉我。
希望能帮助到你。
干杯。