qApp与QApplication.instance()

时间:2016-11-03 12:01:12

标签: python pyqt instance pyqt5 qapplication

使用PyQt5,这两个都返回应用程序对象:

app = QtWidgets.QApplication.instance()
app = QtWidgets.qApp
for i in app.arguments()[1:]:
    ...

但为什么print(QtWidgets.QApplication.instance() is QtWidgets.qApp)会打印False

1 个答案:

答案 0 :(得分:19)

QtWidgets.QApplication.instance()QtWidgets.qApp之间的区别在于后者是静态模块变量,必须在首次导入模块时创建。这导致以下最初令人困惑的行为:

>>> from PyQt5 import QtWidgets
>>> inst = QtWidgets.QApplication.instance()
>>> qapp = QtWidgets.qApp
>>> (inst, qapp)
(None, <PyQt5.QtWidgets.QApplication object at 0x7ff3c8bd3948>)

因此,即使尚未创建QApplication个对象,qApp变量仍指向QApplication个实例。如果模块更像是类,那么它们可以具有动态属性,qApp可以像QApplication.instance()一样工作,并且最初返回None。但是因为它是静态的,所以它必须总是返回一个正确类型的对象,以便稍后可以引用与QApplication.instance()相同的底层C ++对象。

但是,重要的是要注意qApp最初只是一个包装器:

>>> qapp.objectName()
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
RuntimeError: wrapped C/C++ object of type QApplication has been deleted

创建QApplication后,他们都会指向同一件事:

>>> app = QtWidgets.QApplication([])
>>> app.setObjectName('foo')
>>> qapp.objectName()
'foo'

因此(QtWidgets.QApplication.instance() is QtWidgets.qApp)返回False的原因是两个对象是围绕同一底层C ++对象的不同python包装器。

如果您需要创建自己的QApplication子类但仍想使用qApp,请务必注意这一点:

>>> from PyQt5 import QtWidgets
>>> class MyApp(QtWidgets.QApplication):
...     def hello(self): print('Hello World')
...
>>> myapp = MyApp([])
>>> myapp.hello()
Hello World
>>>
>>> QtWidgets.qApp
<PyQt5.QtWidgets.QApplication object at 0x7f5e42f40948>
>>> QtWidgets.qApp.hello()
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
AttributeError: 'QApplication' object has no attribute 'hello'
>>>
>>> inst = QtWidgets.QApplication.instance()
>>> inst
<__main__.MyApp object at 0x7f5e42f409d8>
>>> inst.hello()
Hello World

解决这个问题的唯一方法是显式覆盖qApp模块变量(显然确保在其他模块导入之前完成此操作):

>>> QtWidgets.qApp = myapp
>>> QtWidgets.qApp.hello()
Hello World