QString::toStdString() 在 std::string 析构函数上崩溃

Posted

技术标签:

【中文标题】QString::toStdString() 在 std::string 析构函数上崩溃【英文标题】:QString::toStdString() crashes on std::string destructor 【发布时间】:2013-03-14 17:01:17 【问题描述】:

我已经调试了 2 个小时,归结为这个。如果我打电话给QString::toStdString

QString s = "testtesttesttesttesttest";
const std::string &temp = s.toStdString();

程序稍后在 std::string 析构函数上崩溃

__CLR_OR_THIS_CALL ~basic_string()
       // destroy the string
    _Tidy(true); // <---- It crashes on this line.
    

一开始我以为是内存损坏,但即使main() 只包含那两行,也会发生这种情况。有谁知道为什么会发生这种情况,以及我该如何解决?

我的 Qt 版本是4.8.1

【问题讨论】:

您使用的是什么版本的 Qt?如果您存储字符串 string temp = s.toStdString(); 的副本而不是引用,会发生什么? string temp = s.toStdString(); 也会发生同样的事情,只是发生得更早,因为toStdString() 返回的临时变量的析构函数被立即调用。如何查看我正在使用的 Qt 版本?我在哪里看?该项目已由其他人建立,我没有在 Visual Studio 中使用 Qt 的经验,仅使用 Qt Creator。 试试这个:std::string stdUtf8 = qs.toUtf8().constData(); 同样的事情也发生在我身上!最后我只是调用了 toLocal8Bit。 @Najzero 是的,这行得通。但是我不愿意让这个 bug 潜伏,因为很明显项目的配置有问题。 【参考方案1】:

您的 Qt DLL 需要使用 STL 支持和与您的代码完全相同的 C-Runtime 库进行编译。看起来好像您同时使用了两个不同的 CRT,这会将 Qt 在一个堆上创建的对象销毁到您的程序使用的堆中。

使用 Dependency Walker 检查 DLL 使用情况!

【讨论】:

我打开了我的 .exe 和 QtCored4.dll 但我不确定要查找什么。他们有很多依赖。我怎么知道他们是否使用不同的 CRT?不同的名字,还是别的什么? 只要确保您使用与您的应用程序相同的编译器编译 Qt。还要确保你没有混合调试和发布。 从 DependencyWalker(“Profiling”)启动您的应用程序,然后查看加载了多少 MS-CRT。如果不止一个,你很有可能会遇到麻烦。 这将发现不同的动态 CRT。 如果加载了任何 msvc 运行时库的多个版本,我建议在调试器中查找。如果您使用 Visual Studio 启动(或附加)流程,请打开模块窗口。它列出了所有加载的 DLL 和可执行文件。如果它列出了多个此类 DLK,这可能就是原因。 DependencyWalker 很有用,但也不可靠。真正可怕的是我们决定使用哪些 Dll 的 Windows SideBySide 管理器。【参考方案2】:

最可能的原因可能是您的运行时库是“多线程 (/MT)”,您需要将其更改为“多线程 DLL (/MD)”(如果您使用的是发布版本)

如果您使用的是调试版本,请从“多线程调试 (/MTd)”更改为“多线程调试 DLL (/MDd)”

如果你有一个奇怪的 Qt 编译,解决方案应该是相反的。

您会在“配置属性->C/C++->代码生成->运行时库”中找到它

【讨论】:

【参考方案3】:

我尝试以不同的方式解决问题。我从 Visual Studio 创建了一个新项目,并且测试代码没有在那里崩溃。在使用 WinMerge 检查 *.vcproj 文件之间的差异后,我发现崩溃是由项目中的一些自定义更改引起的 - 你猜对了 - 运行时库。这是 WinMerge 创建的补丁,其中包含导致崩溃重现的最小差异:

112c112
<               RuntimeLibrary="3"
---
>               RuntimeLibrary="1"
126a127,128
>               LinkLibraryDependencies="true"
>               UseLibraryDependencyInputs="false"
127a130,131
>               IgnoreAllDefaultLibraries="false"
>               IgnoreDefaultLibraryNames="msvcrtd.lib"

【讨论】:

以上是关于QString::toStdString() 在 std::string 析构函数上崩溃的主要内容,如果未能解决你的问题,请参考以下文章

Qt 4.8.2 QString toStdString 未处理异常

损坏的 winapi 可执行清单

如何使用 msvc 2019 编译 QT 4 dll?

分配的变量引用在哪里,在堆栈中还是在堆中?

NOIP 2015 & SDOI 2016 Round1 & CTSC 2016 & SDOI2016 Round2游记

秋的潇洒在啥?在啥在啥?