2

auto_ptr リリース メンバーを呼び出すと、コードのこの時点でセグメンテーション エラーが発生します。

try
{
    newMod->init(params);
}
catch (const std::exception& e)
{
#ifndef CONFIG_STATIC
    dlclose(handle);
#endif
    throw std::runtime_error(utils::buildString(
            "%s: Error initializing module %s: %s",
            DBG_FUNC_NAME, newMod->name().c_str(), e.what()));
}

_modules.insert(std::make_pair(newMod->name(), newMod.release()));

_modules の場所

std::map<std::string, IModule*> _modules;

そしてnewModは

std::auto_ptr<IModule> newMod(0);

後で適切なポインター値でリセットします。リリース前に init メンバーを呼び出しているため、IModule へのポインターが有効であることはわかっています。

これ:

_modules.insert(std::make_pair(newMod->name(), newMod.get()));
newMod.release();

完全にうまく機能し、これは gdb が言うことです:

#0  _M_rep (this=0xbfe8e908, __str=...) at /usr/src/debug/gcc-4.5.1-20101208/obj-i586-suse-linux/i586-suse-linux/libstdc++-v3/include/bits/basic_string.h:287
#1  std::basic_string<char, std::char_traits<char>, std::allocator<char> >::basic_string (this=0xbfe8e908, __str=...)
at /usr/src/debug/gcc-4.5.1-20101208/obj-i586-suse-linux/i586-suse-linux/libstdc++-v3/include/bits/basic_string.tcc:173
#2  0x0805d76f in sm::core::mod::ModuleManager::loadModule (this=0x8074768, name=..., params=...) at src/core/ModuleManager.cpp:150
#3  0x08056edb in sm::core::Main::start (this=0xbfe8e9e0) at src/core/Main.cpp:82
#4  0x08055131 in main (argc=4, argv=0xbfe8ebf4) at src/core/smmain.cpp:15

セグメンテーション違反の行は次のとおりです。

{ return &((reinterpret_cast<_Rep*> (_M_data()))[-1]); }

これで何が間違っている可能性がありますか?

4

1 に答える 1

4
_modules.insert(std::make_pair(newMod->name(), newMod.release()));

関数への引数の評価の順序は、標準では定義されていません。実装はnewMod.release()beforeを評価できnewMod->name()ます。これにより、2番目の呼び出しが無効になります。


補足:auto_ptrC ++ 11では非推奨です(ドラフトn3290の付録D.10):

クラステンプレートauto_ptrは非推奨になりました。[注:クラステンプレートunique_ptr(20.7.1)は、より優れたソリューションを提供します。—エンドノート]

これにアクセスでき、そのための時間/リソースがある場合は、新しいスマートポインタークラスに切り替えることを検討してください。

于 2012-05-05T08:35:47.987 に答える