我喜欢 python 和 Qt,但对我来说很明显 Qt 在设计时并没有考虑到 python。有许多方法可以使 PyQt / PySide 应用程序崩溃,其中许多方法都非常难以调试,即使使用适当的工具也是如此。
我想知道:在使用 PyQt 和 PySide 时,避免崩溃和锁定的良好做法是什么?这些可以是任何东西,从一般的编程技巧和支持模块到高度具体的解决方法和要避免的错误。
请注意 Qt 自动删除对象的情况。如果 python 包装器没有被告知 C++ 对象已被删除,那么访问它会导致崩溃。由于 PyQt 和 PySide 在跟踪 Qt 对象方面存在困难,这可能以许多不同的方式发生。
从 QTreeWidget 中删除项目将导致任何关联的小部件(使用 QTreeWidget.setItemWidget 设置)被删除。
# Example:
from PyQt4 import QtGui, QtCore
app = QtGui.QApplication([])
# Create a QScrollArea, get a reference to one of its scroll bars.
w = QtGui.QWidget()
sa = QtGui.QScrollArea(w)
sb = sa.horizontalScrollBar()
# Later on, we delete the top-level widget because it was removed from the
# GUI and is no longer needed
del w
# At this point, Qt has automatically deleted all three widgets.
# PyQt knows that the QScrollArea is gone and will raise an exception if
# you try to access it:
sa.parent()
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
RuntimeError: underlying C/C++ object has been deleted
# However, PyQt does not know that the scroll bar has also been deleted.
# Since any attempt to access the deleted object will probably cause a
# crash, this object is 'toxic'; remove all references to it to avoid
# any accidents
sb.parent()
# Segmentation fault (core dumped)
只是补充一点:
如果您必须在基于 qt 的程序中使用线程,您确实必须禁用自动垃圾收集器并在主线程上进行手动收集(如http://pydev.blogspot.com.br/2014/03/should- python-garbage-collector-be.html)-请注意,即使您确保对象没有循环,也应该这样做(通过循环,您基本上可以使对象存活,直到 python 循环垃圾收集器出现,但是有时,如果您有例外情况,则可能会保持帧处于活动状态,因此,在这种情况下,您的对象可能仍然保持活动的时间比您预期的要长)...在这些情况下,垃圾收集器可能会遇到问题在辅助线程中,这可能导致 qt 段错误(qt 小部件必须始终收集在主线程中)。
仅供参考,我将PyQt 的作者对Luke 的回答发表评论:“这是垃圾。”
我认为这很重要,因为有人可以跳入这篇文章并仍然对所有这些(不存在的)“问题”感到困惑。
发出信号而不是同步 UI 控制是我在实现逻辑电路模拟器时避免问题的关键