专栏首页24K纯开源Qt程序继承QApplication发生崩溃的原因

Qt程序继承QApplication发生崩溃的原因

一、前情介绍

QApplication是Qt开发中经常用到的一个类,用来管理应用程序的生命周期。跟其相关的类还有QCoreApplication和QGuiApplication,分别用于不同场景下为应用程序的控制流和事件处理提供基础的框架。这三个类的构造函数都接收两个参数(分别是argc和argv),和C/C++程序的main函数的参数差不多。因此,大部分情况下我们是直接将main函数的这两个参数传给QApplication(这里以GUI程序为例):

1 #include <QApplication>
2 
3 int main(int argc, char *argv[])
4 { 
5     QApplication app( argc, argv );
6     // Create main window...
7 
8     return app.exec();
9 }

绝大部分情况下这不会有什么问题,程序能够正常运行并结束。但是最近遇到的一个Qt程序崩溃的问题,却不得不让我对QApplication的两个参数提高了警惕。情况是这样的,我们在项目中为了保存一些全局性的数据,从QApplication派生了一个子类,并增加了一些新的方法来保存运行时的数据。编译运行很开心,程序完全满足了我们的要求。但是程序发布出去给用户使用的时候,我们在后台的崩溃上报系统中看到了一个这样的崩溃堆栈:

很明显程序在QCoreApplication的arguments()方法中崩溃了。这个崩溃堆栈让我们不由得浮想联翩:难道这个是Qt框架本身的Bug?不小心被我给踩到了?因为我们的程序运行起来之后,没有什么地方会和QCoreApplication的arguments方法打交道啊!这么一想心里顿时好受多了,帅锅技能升华!

过了一段时间之后,另外一个同事想在mac电脑上来编译工程,却发现编译后的程序死都运行不起来。一运行就报错:EXC_i386_GPFLT QCoreApplication::arguments,又将矛头指向了QCoreApplication的arguments方法,这下我慌了!这下必须要仔细排查下原因,不能假装不知道继续帅锅了!根据关键字EXC_i386_GFLT没用找到什么有用的东西,再一搜Qt QApplication arguments方法崩溃,就找到了一堆的信息,其中Qt bug管理系统上的一个用户吐槽最为详细:

这个用户说的很详细,QApplication的构造函数中argc必须为引用传值方式,否则程序会崩溃!然而Qt官方文档并没有强调这一点,导致很多用户根本没在意到这一点。再去看Qt文档,可以发现QApplication,QCoreApplication和QGuiApplication的构造函数中,argc都是引用传值的方式声明的。确实粗心大意了!

二、参考链接

1. https://bugreports.qt.io/browse/QTBUG-5637

2. https://groups.google.com/forum/#!msg/mathgl/CVvVyRnl3X4/EPiRrnqy3moJ

3. https://stackoverflow.com/questions/22243674/segmentation-fault-when-qt-qapplication-created-with-new/22245111

4. https://stackoverflow.com/questions/35566459/segfault-when-accessing-qapplicationarguments

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • 集成Qt Webkit 到cocos2d-x

    近期倒腾下客户端,想搞个cocos2d的工具。 之前的那个集成到Win32工具下的调试辅助工具是直接用的windows api。拓展起来巨麻烦。而且Windo...

    owent
  • Qt项目之高亮关键字Python编辑器实现

    之前大部分内容在写Qt一些小部件以及基础模块的用法,不成体系,大部分时候还是用什么找什么。随着对Qt的逐渐熟悉,应该做一些项目,这样可以在实际应用的过程中加深对...

    用户5908113
  • 从零开始用 PyQt5 写一个 scihub 下载器(二)

    在上一次的教程中,我们已经设计了程序界面,并生成了界面的 .py 脚本。在今天的教程中,我们将介绍如何使用这种界面与逻辑分离的 GUI 程序框架,构建主函数,并...

    生信菜鸟团
  • Qt插件入门——使用插件扩展Qt自身

    用于编写Qt自身扩展的高级API:自定义数据库驱动程序,图像格式,文本编解码器,自定义样式等。本次的例子就是自定义样式的例子。

    用户5908113
  • PySide——Python图形化界面入门教程(一)

    PySide——Python图形化界面入门教程(一) ——基本部件和HelloWorld 翻译自:http://pythoncentral.io/intro-...

    ascii0x03
  • python Socket网络编程实现C/S模式和P2P

    由于网络课需要实现Socket网络编程,所以简单实现了一下,C/S模式分别用TCP/IP协议与UDP协议实现,下面将分别讲解。

    砸漏
  • 使Qt程序只能运行一个实例的3种方法

    Unix: QSharedMemory "owns" the shared memory segment. When the last thread or pr...

    用户5807183
  • 19.QT-事件发送函数sendEvent()、postEvent()

    张诺谦
  • 19.QT-事件发送函数sendEvent()、postEvent()

    张诺谦

扫码关注云+社区

领取腾讯云代金券