BOOST應用 無法解析的外部符號 "void __cdecl boost::throw_exception(class std::exception const &)"

劉近光發表於2018-03-15

簡介

在Windows系統上,使用boost時發現未定義的外部符號的連結問題:

thread_test_01.cpp.obj : error LNK2019: 無法解析的外部符號 "void __cdecl boost::
throw_exception(class std::exception const &)" (?throw_exception@boost@@YAXAEBVe
xception@std@@@Z),該符號在函式 "public: __cdecl boost::gregorian::date::date(cl
ass boost::gregorian::greg_year,class boost::gregorian::greg_month,class boost::
gregorian::greg_day)" (??0date@gregorian@boost@@QEAA@Vgreg_year@12@Vgreg_month@1
2@Vgreg_day@12@@Z) 中被引用
thread_test_01.exe : fatal error LNK1120: 1 個無法解析的外部命令
本文總結了該問題的解決思路。

問題分析

搜尋程式碼,發現boost/throw_exception.hpp檔案中定義瞭如下相關的程式碼:

#ifdef BOOST_NO_EXCEPTIONS

void throw_exception( std::exception const & e ); // user defined

#else

inline void throw_exception_assert_compatibility( std::exception const & ) { }

template<class E> BOOST_NORETURN inline void throw_exception( E const & e )
{
    //All boost exceptions are required to derive from std::exception,
    //to ensure compatibility with BOOST_NO_EXCEPTIONS.
    throw_exception_assert_compatibility(e);

#ifndef BOOST_EXCEPTION_DISABLE
    throw enable_current_exception(enable_error_info(e));
#else
    throw e;
#endif
}

#endif

由於定義了BOOST_NO_EXCEPTIONS巨集,使用者需要自己實現throw_exception函式。進一步搜尋BOOST_NO_EXCEPTIONS巨集,發現程式碼中相關定義如下:

#if !defined(_CPPUNWIND) && !defined(__EXCEPTIONS)
#  define BOOST_NO_EXCEPTIONS
#endif

而巨集_CPPUNWIND用於標識編譯器是否開啟異常處理。而這裡需要開啟異常處理的功能,在Visual studios中,需要在編譯選項中使用/GX或者/EHsc,告訴編譯器使能異常處理的功能。如果使用cmake,可以使用下面的設定來進行配置:

set(CMAKE_CXX_FLAGS "/EHsc ${CMAKE_CXX_FLAGS}")

相關文章