使用语言环境时 C++ boost 崩溃

标签 c++ boost crash locale

我正在尝试使用 boost 库为我的字符串类提供 i18 支持。 我正在使用 Microsoft Visual Studio 编译器 VC10 和 64 位 Windows 7 机器。

我能够编译我的应用程序并将其与 boost 库链接,但是我的应用程序在调用 boost::locale::to_upper() 时崩溃。

下面是我写的代码。

#include <boost/locale.hpp>
#include <boost/algorithm/string/case_conv.hpp>
#include <boost/system/config.hpp>



 String::MakeUpper()()
    {
    boost::locale::generator gen;
    std::locale loc = gen("");
    std::locale::global(loc);
    std::string str2 = boost::locale::to_upper("001Öä", loc); // application crashes here.
    std::string str3 = boost::locale::to_upper("001Öä"); // This also does not work
    }

崩溃发生在以下函数中。此函数抛出错误的转换异常。

template<class _Facet> inline
const _Facet& __CRTDECL use_facet(const locale& _Loc)

{   // get facet reference from locale
_BEGIN_LOCK(_LOCK_LOCALE)   // the thread lock, make get atomic
    const locale::facet *_Psave =
        _Facetptr<_Facet>::_Psave;  // static pointer to lazy facet

    size_t _Id = _Facet::id;
    const locale::facet *_Pf = _Loc._Getfacet(_Id);

    if (_Pf != 0)
        ;   // got facet from locale
    else if (_Psave != 0)
        _Pf = _Psave;   // lazy facet already allocated
    else if (_Facet::_Getcat(&_Psave, &_Loc) == (size_t)(-1))

#if _HAS_EXCEPTIONS

        _THROW_NCEE(bad_cast, _EMPTY_ARGUMENT); // lazy disallowed

....
....
....

}

你能帮帮我吗

问候, 提交

最佳答案

在 Windows 7 64 位上的 Visual Studio 2008 应用程序中使用 Boost 1.55 的静态库构建时,我遇到了同样的问题,其中主要可执行文件和几个 DLL 都链接到 Boost。我不确定你的问题是否和我的一样,因为你没有提到使用 DLL,但当我第一次开始调查这个问题时,我认为这不是相关的。

如果您只是对解决此问题的最直接方法感兴趣,那么将 Boost 构建为共享库应该可以做到。具体来说,我的意思是将 b2 命令行的 link 属性设置为 shared 而不是 static

说明

使用静态库构建有问题的原因是 Boost.Locale 使用 std::locale::facet对象进行文本转换操作,如大写和规范化。 std::locale::facet 类需要有一个 id 静态成员变量,其唯一值由标准库实现在静态初始化期间构造时分配。

使用静态库时的问题是所有可执行文件和 DLL 都从静态库中获取静态成员变量的不同拷贝,如 Shared global variable in C++ static library 中所述。 .当您使用 boost::locale::generator::operator() 生成语言环境时,它只会将 std::locale::facet 对象安装到语言环境中有一个 id 成员变量,它是包含该调用的同一 DLL 或可执行文件的一部分。

正如我上面所说,解决此问题的最直接方法是将 Boost 构建为共享库。这样就只有一份 Boost.Locale 的静态成员变量。具体来说,它们将位于 Boost.Locale DLL 中。

替代方案

您可以使用 Boost 的静态库构建来做到这一点,方法是确保所有 DLL 中的所有 std::locale::facet 对象和使用 Boost.Locale 的可执行文件都安装到 std::locale 您尝试使用的对象。

您可以使用如下代码来执行此操作。对于 DLL,当它的第二个参数 fdwReasonDLL_PROCESS_ATTACH 时,您可以在 DllMain 中调用它,对于您的可执行文件,您可以在 中调用它>WinMain 或其他一些应用程序入口点(如果您使用的是 MFC 或 Qt 之类的东西)。

void setup_global_locale()
{
    const boost::locale::generator generator;
    const std::locale locale = generator.generate( std::locale(), "" );
    std::locale::global( locale );
}

代码的重要部分是它每次运行时都使用全局语言环境作为基本语言环境,并将新生成的语言环境安装为新的全局语言环境。这与 boost::locale::generator::operator() 将执行的操作不同,因为它使用 std::locale::classic 作为基本语言环境,并且那一个不能修改。通过从每个 DLL 和可执行文件调用它,您会将它们的每个 std::locale::facet 对象安装到全局语言环境中。

关于使用语言环境时 C++ boost 崩溃,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/26990412/

相关文章:

c++ - 如何创建模板依赖成员类型

c++ - Boost Karma - 非消耗谓词

c++ - boost asio 示例编译错误

visual-studio-2012 - Visual Studio 2012 打开后崩溃

C++模板鸭子类型与纯虚拟基类继承

c++ - 用 C++ 比较两个 SQLite 数据库

c++ - 池内存和 std::forward_list

c++ - boost ASIO 串行写入十六进制值

Android 在主线程上跳帧过多

objective-c - '-[__ NSCFArray objectAtIndex:]:索引(1)超出范围(0)'崩溃