在 wifstream 析构函数中中止

Abort in wifstream destructor

以下 C++ 代码在执行 return 时出现神秘错误 ("Debug Error!...abort() has been called")。这是 Visual Studio 2017 15.6.3,程序是 64 位调试版本,运行 在 64 位下 Windows 7. 导致此错误的原因是什么,我该如何解决是吗?

wifstream inFile;

std::codecvt_utf16<wchar_t, 0x10ffff, std::little_endian> cv1;
inFile.imbue(std::locale(inFile.getloc(), &cv1));
return 0;

使用调试器进行跟踪显示消息来自反汇编程序指令

call        std::basic_ifstream<wchar_t,std::char_traits<wchar_t> >::`vbase destructor'

除了 msvcp140d.dllvcruntime140d.dllucrtbased.dll 之外,堆栈中的最后一个条目是

    MyApp.exe!std::basic_ifstream<wchar_t,std::char_traits<wchar_t> >::`vbase destructor'() C++

此代码的目的是输入文件 infile 是 Unicode(小端),我正在将其读入 std::string 变量。

std::locale 为与其关联的每个方面维护一个引用计数。您调用的 std::locale 构造函数将增加您传入的 std::codecvt_utf16 对象的引用计数,然后 std::locale 的析构函数将减少该引用计数。当 std::codecvt_utf16 的引用计数降为 0 时,它将通过 delete 运算符销毁。这就是为什么您会收到中止错误的原因 - 当 std::wifstream 析构函数正在清理 imbue 的语言环境时,该语言环境的析构函数会尝试 delete 未分配给 new 运算符开头。

改为这样做:

inFile.imbue(std::locale(inFile.getloc(),
    new std::codecvt_utf16<wchar_t, 0x10ffff, std::little_endian>));

参见 std::codecvt_utf16 documentation on cppreference.com 中的示例。