Qt5 抛出 std::bad_alloc

Posted

技术标签:

【中文标题】Qt5 抛出 std::bad_alloc【英文标题】:Qt5 throws std::bad_alloc 【发布时间】:2018-10-04 04:06:27 【问题描述】:

我正在尝试在我的控制台应用程序中使用 QCustomPlot。我首先为它创建了一个适合我使用的简单包装器。包装器应该是但是,每次我尝试显示窗口时都会遇到 std::bad_alloc 错误。

这是我的代码,我在Plot.hpp 中创建了一个包装类:

class Plot

    private:
        std::string name;
        QApplication* app;
        QMainWindow* window;
        QCustomPlot* plotWidget;
    public:
        Plot(std::string& name);
        // OTHER METHODS
        void showPlot();
;

在我的Plot.cpp 文件中,我有以下内容:

Plot::Plot(std::string& name) : name(name)

    char *gui_argv[] = (char*)(name.c_str()), NULL;
    int gui_argc = sizeof(gui_argv) / sizeof(char*) - 1;
    app = new QApplication(gui_argc, gui_argv);
    window = new QMainWindow();
    // Add plot Widget
    plotWidget = new QCustomPlot(window);
    window->setCentralWidget(plotWidget);
    plotWidget->plotLayout()->clear();


// OTHER METHODS

void Plot::showPlot()

    // Run the GUI
    window->show();
    app->exec();

我的main.cpp 中有以下内容:

int main()

    std::string title = "Testing";
    Util::Plot *plotWindow = new Util::Plot(title);
    // NO OTHER STATEMENTS
    plotWindow->showPlot();
    return 0;

通过 GDB,我得到了这个堆栈跟踪,但我无法真正破译它以找出错误所在。它深入到 QT 的内部:

#0  0x00007ffff7279e97 in __GI_raise (sig=sig@entry=6) at ../sysdeps/unix/sysv/linux/raise.c:51
#1  0x00007ffff727b801 in __GI_abort () at abort.c:79
#2  0x00007ffff78d08fb in  () at /usr/lib/x86_64-linux-gnu/libstdc++.so.6
#3  0x00007ffff78d6d3a in  () at /usr/lib/x86_64-linux-gnu/libstdc++.so.6
#4  0x00007ffff78d6d95 in  () at /usr/lib/x86_64-linux-gnu/libstdc++.so.6
#5  0x00007ffff78d6fe8 in  () at /usr/lib/x86_64-linux-gnu/libstdc++.so.6
#6  0x00007ffff529e402 in  () at /usr/lib/x86_64-linux-gnu/libQt5Core.so.5
#7  0x00007ffff530a22a in QListData::detach(int) () at /usr/lib/x86_64-linux-gnu/libQt5Core.so.5
#8  0x00007ffff534475e in  () at /usr/lib/x86_64-linux-gnu/libQt5Core.so.5
#9  0x00007ffff549a48f in QCoreApplication::arguments() () at /usr/lib/x86_64-linux-gnu/libQt5Core.so.5
#10 0x00007fffef9e3791 in  () at /usr/lib/x86_64-linux-gnu/libQt5XcbQpa.so.5
#11 0x00007fffef9e3c8d in QXcbIntegration::wmClass() const () at /usr/lib/x86_64-linux-gnu/libQt5XcbQpa.so.5
#12 0x00007fffef9f8e03 in QXcbWindow::create() () at /usr/lib/x86_64-linux-gnu/libQt5XcbQpa.so.5
#13 0x00007fffef9e4bfb in QXcbIntegration::createPlatformWindow(QWindow*) const () at /usr/lib/x86_64-linux-gnu/libQt5XcbQpa.so.5
#14 0x00007ffff5a6229e in QWindowPrivate::create(bool, unsigned long long) () at /usr/lib/x86_64-linux-gnu/libQt5Gui.so.5
#15 0x00007ffff6245add in QWidgetPrivate::create_sys(unsigned long long, bool, bool) () at /usr/lib/x86_64-linux-gnu/libQt5Widgets.so.5
#16 0x00007ffff624619d in QWidget::create(unsigned long long, bool, bool) () at /usr/lib/x86_64-linux-gnu/libQt5Widgets.so.5
#17 0x00007ffff6252a96 in QWidget::setVisible(bool) () at /usr/lib/x86_64-linux-gnu/libQt5Widgets.so.5
# The following is line window->show()
#18 0x00007ffff7bd3179 in Util::Plot::showPlot() (this=0x55555576fd80) at ./lib/util/Plot.cpp:71
#19 0x00005555555549b3 in main() () at ./lib/test/PlotTest.cpp:16

我还验证了指向windowappplotWidget 的指针不为空。所以基本上,只需创建QMainWindow 并尝试显示它,甚至不执行任何其他操作都会导致此故障发生。这里有什么问题?我错过了什么?

额外: 我不认为以下是问题的原因。但以防万一: 我没有使用 QT Studio,我已经编写了自己的 makefile 来构建 libQCustomPlot.so 和我自己的应用程序,并将它们与必要的 QT 库链接起来。编译过程中没有失败或警告。

编辑1: 我忘了发布原始错误!只是以下内容,没有其他信息/说明:

terminate called after throwing an instance of 'std::bad_alloc'
  what():  std::bad_alloc
Aborted (core dumped)

【问题讨论】:

【参考方案1】:

QApplication 通过引用获取argc,并希望此引用在应用程序的生命周期内有效。一旦您的 Plot 函数结束,QApplication 就会留下对gui_argc 的悬空引用,因此当调用 QApplication::arguments (如您的回溯中所见)时,会发生未定义的行为。您可以通过在某处持久化 argc 来解决此问题。

【讨论】:

是的,for reference:警告:argcargv 引用的数据必须在 QApplication 对象的整个生命周期内保持有效。 问题出现在我的代码中,看起来好像是竞争条件。无害的代码更改似乎可以暂时修复它。感谢您的解决方案!

以上是关于Qt5 抛出 std::bad_alloc的主要内容,如果未能解决你的问题,请参考以下文章

在 Ubuntu 14.04 上安装 Qt 5.5.1 - 在终端上抛出 qt.network.ssl 错误

如何在Qt中捕获异常?

树莓派上的 Qt OpenGL 上下文

Qt5.5.1和Qt5.3.2编译OCI驱动教程及验证方法

Qt5怎样使用OpenGL

由于找不到qt5webenginecore.dll无法继续执