使用PyQt5,这两个都返回应用程序对象:
app = QtWidgets.QApplication.instance()
app = QtWidgets.qApp
for i in app.arguments()[1:]:
...
但为什么print(QtWidgets.QApplication.instance() is QtWidgets.qApp)
会打印False
?
答案 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