- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我想知道在同一个进程中有多个 QApplication
/QCoreApplication
实例有什么影响(问题),以及如何解决一些相关问题.
场景如下:我想在开源第三方应用程序上制作一个包装器,以便将其转换为可嵌入的小部件作为可选插件(该应用程序基本上由单个 QMainWindow 组成基于
的接口(interface))。
此类项目严重依赖于 QCoreApplication
派生类,但主要是因为它被用作已经存在的单例。我能够修改代码(我必须这样做才能将 QMainWindow
公开为可嵌入的小部件),尽管由于该项目的复杂性,我不能简单地删除 父类。
因此,最终应用程序将拥有自己的QApplication
(在启动时创建),然后可选择加载上述插件(从而创建第二个QCoreApplication
)。只有第一个(主)QApplication
用于事件循环(QCoreApplication::exec()
)。
我知道 QCoreApplication
是一个单例。在我的测试中,单例总是指向最后创建的实例:
qDebug() << qApp;
auto app1 = new QApplication(argc, argv);
qDebug() << qApp;
auto app2 = new TheOtherQApplication(argc, argv);
qDebug() << qApp;
输出是
QObject(0x0)
QApplication(0x6f9400, name = "test")
ASSERT failure in QCoreApplication: "there should be only one application object", file kernel\qcoreapplication.cpp, line 595
TheOtherQApplication(0x2550dc0, name = "test")
TheOtherQApplication(0x2550dc0, name = "test") TheOtherQApplication(0x2550dc0, name = "test")
可以看出,在创建第二个QApplication
后,它替换了全局实例。有什么办法可以解决这个问题吗?由于该插件是可选的,因此显而易见的答案(加载主要的 QApplication
在第二位)不是一个合适的选择。
此外,拥有多个 QApplication
实例是否还有其他影响?还是都和事件循环(勾选)和单例有关?
注意:由于第三方依赖尚未完全更新,这是一个基于 Qt 4.7 的项目。计划在一年左右迁移到最新版本,但目前我必须处理 4.7。
顺便说一句,我已经回顾了几个相关问题,包括 this one但这并没有提供任何有用的信息。
最佳答案
嗯,据我所知,使用两个或多个Q*Application
(QCoreApplication
、QGuiApplication
、 QApplication
) 意味着:
创建第二个(或更多)应用程序时,断言在 Debug模式下失败。 Release模式下没有崩溃。
Q*Application
的每个实例都会更新单例(即,qApp
将始终指向最后一个实例)。
应用程序名称和版本等全局属性随实例传输并覆盖之前的属性。
连接到 Q*Application
槽的任何信号都会调用单例槽,即使在创建最新实例之前已连接也是如此。
只有连接到最新 Q*Application
信号的槽才会被调用(它们不会转移到新实例)。
创建新的 Q*Application
实例时,不会传输翻译器。
如果最后一个 Q*Application
被销毁,则单例变为 null(它不会回退到前一个实例)。
您可以使用以下代码并切换 USE_TWO_QAPPS
来测试这些功能:
#include <QtCore>
#define USE_TWO_QAPPS
int main(int argc, char* argv[])
{
QTranslator tr1;
QCoreApplication a1(argc, argv);
a1.setApplicationName("a1");
a1.installTranslator(&tr1);
qDebug() << qApp << &a1;
qDebug() << "a1.applicationName() =" << a1.applicationName();
// qApp == &a1
QObject::connect(&a1, &QCoreApplication::aboutToQuit, []() {
// point 5, never called with Q*Application
qDebug() << "Hello world from a1!";
});
QTimer::singleShot(2000, &a1, &QCoreApplication::quit); // as if connected to latest qApp, point 4
#ifdef USE_TWO_QAPPS
// if (true) { // uncomment to test point 7
QCoreApplication a2(argc, argv);
a2.setApplicationName("a2");
qDebug() << qApp << &a1 << &a2; // test point 2
qDebug() << "a2.applicationName() =" << a2.applicationName();
qDebug() << "a1.applicationName() =" << a1.applicationName(); // as if called from qApp, point 3
QObject::connect(&a2, &QCoreApplication::aboutToQuit, []() {
qDebug() << "Hello world from a2!";
});
// } // uncomment to test point 7
#endif
qDebug() << qApp->removeTranslator(&tr1); // false if the translator is not installed, point 6
a1.installTranslator(&tr1); // it is installed in the latest instance (as if called from qApp)
qDebug() << qApp->removeTranslator(&tr1);
return qApp->exec();
}
一个Q*Application
的结果
QCoreApplication(0xfafb74) QCoreApplication(0xfafb74)
a1.applicationName() = "a1"
true
true
Hello world from a1!
两个Q*Application
的结果
QCoreApplication(0xbefb2c) QCoreApplication(0xbefb2c)
a1.applicationName() = "a1"
ASSERT failure in QCoreApplication: "there should be only one application object", file ########\qtbase\src\corelib\kernel\qcoreapplication.cpp, line 769
QCoreApplication(0xbefb1c) QCoreApplication(0xbefb2c) QCoreApplication(0xbefb1c)
a2.applicationName() = "a2"
a1.applicationName() = "a2"
false
true
Hello world from a2!
在测试点 7 时,a2
在退出 if
语句时被销毁。在这种情况下,每次调用 Q*Application
方法都会引发警告并且不会执行(它们不会崩溃并且没有断言被破坏)。即使从以前的应用程序调用时也会发生这种情况:a1.installTranslator(&tr1);
QApplication::installTranslator: Please instantiate the QApplication object first
注意:使用 Visual Studio 2010 进行测试。Qt 版本为 4.7 和 5.6.1-1,两者的结果相同
更新:https://github.com/cbuchart/stackoverflow/blob/master/46304070-multiple-qapplication-instances/main.cpp 中提供了此答案的更简洁代码版本
在评论之后,这段代码还测试了当所有 QApplication
对象被销毁然后再次创建时会发生什么。结果:正如预期的那样,没有什么特别的发生,似乎没有副作用。
结论
考虑到这些要点,似乎可以使用两个或多个 Q*Application
,更重要的是,除了最后一个 Q*Application
,与信号的连接丢失并且未安装翻译器。此外,如果最后一个实例被销毁,则没有应用程序可用,因此您应该注意这些情况(例如,如果卸载创建最后一个实例的 DLL)。
关于c++ - 多个 QApplication 实例,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/46304070/
我想念任何Qt如果我替换 QApplication::exec() 的功能使用标准的 Windows 消息循环实现?这应该澄清我的意思: 运行事件处理的常用“Qt”方式: int main(int a
何时在 Windows 上调用虚函数 QApplication::saveState 和 QApplication::commitData?它们是否被调用,或者 session 管理只是 UNIX 的
有两个qApplications A & B,它们可以在各自的主窗口中分别执行。 我想实现以下目标: 1) //Open Application B. //Inside App B's code
我正在尝试运行一个简单的 hello world 示例,并且已经需要一些时间来弄清楚要使用的内容现在我验证了包含路径,QApplication 实际上应该在那里,但它抛出了上述错误。为了清楚起见,我的
当我首先运行 python-pyside2 项目服务器时,它运行良好。 而且该站点也运行良好,但是如果我按 F5 btn 在浏览器中刷新。 站点显示错误页面 Runtime at/ import sy
关闭。这个问题是not reproducible or was caused by typos .它目前不接受答案。 这个问题是由于错别字或无法再重现的问题引起的。虽然类似的问题可能是on-topi
我正在将 Qt 桌面应用程序移植到 Linux(Ubuntu 19.10、64 位桌面、Qt 5.12.5、gcc 版本 9.2.1),并且在 QApplication 完成后发现一些意外的线程仍然处
这里引用 Qt 文档: For any GUI application using Qt, there is precisely one QApplication object 很明显。但我不知道为什
这里引用 Qt 文档: For any GUI application using Qt, there is precisely one QApplication object 很明显。但我不知道为什
我正在尝试将修改后的 qt 程序作为库运行。我需要它而不是阻止主执行。 所以我想做的是运行 QApplication 并开始执行主应用程序。我该如何实现? 我的第一个想法是在一个单独的线程中运行它。
我需要在单击按钮时重新启动我的应用程序,但我遇到了一些问题。我尝试了两种方法: 已尝试 this suggestion它确实重新启动了应用程序,但是我得到了每个小部件的 Gtk_IS_INVISIBL
我试图在不同的线程中创建 QApplication,但发现了 2 个主要问题: 1- 我无法与 GUI 交互 2- 一些警告: 警告:QApplication 不是在 main() 线程中创建的。 Q
是否可以在退出前一个实例后创建和使用新的 QApplication 实例? 最佳答案 是的,您可以在销毁前一个实例后创建一个新的 QApplication。我在 Windows 中使用 PyQt4 验
我是 PyQt4(和 QT)的新手,我遇到了一个问题, 我已经对 QApplication 进行了子类化(以具有对应用程序真正全局的全局数据和函数): class App(QApplication):
我正在 3Dsmax 2015 上做一些简单的 PySide。 这是我的错误: python.ExecuteFile "C:\Program Files\Autodesk\3ds Max 2015\s
我有一个带有单个对话框窗口的 Qt 应用程序。主要看起来像这样: QApplication a(argc, argv); MyObject * myMobject = new QObject(&a);
我有一个带有通知区域图标的应用程序,因此主窗口可能会忽略关闭事件。我正在尝试在应用程序退出时保存首选项和历史记录。我还想在应用程序运行时处理注销。虽然该应用程序是跨平台的,但它在 GNU/Linux
QApplication::QApplication ( int & argc, char ** argv ) Initializes the window system and constructs
我应该在哪里使用 except 块来记录 QApplication 的异常? 这似乎不起作用: app = QtGui.QApplication(sys.argv) MainWindow = Main
在基本的 C++ Qt Widgets 应用程序中,在堆栈上创建一个 QApplication 或直接在 main 函数中对其进行堆分配是可行的,但是在堆上调用一个函数-分配一个 QApplicati
我是一名优秀的程序员,十分优秀!