我遇到的问题是非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中实现解决方案,即保持对象通用以用作其他程序中的库。
感谢您的帮助。
答案 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)
它被垃圾收集,因为一旦函数执行结束,范围就会丢失。您需要保存引用,有两种方法:
QWidget
实现中,将创建的对象保存在widget对象中(通过self)。如果我不清楚或不够准确,请告诉我。
希望能帮助到你。
欢呼声。