我有一个随Visual Studio 2019打开的CMake项目。我的代码需要c ++ 17功能,因此我在CMakeLists.txt
cmake_minimum_required (VERSION 3.10.0)
project (datalog)
message (STATUS "Building project ${PROJECT_NAME}")
find_package(stxxl CONFIG REQUIRED)
include_directories (${CMAKE_SOURCE_DIR}/src)
set (PROJECT_SRC
main.cpp
)
add_executable (${PROJECT_NAME} ${PROJECT_SRC})
target_compile_features(${PROJECT_NAME} PRIVATE cxx_std_17)
target_link_libraries(${PROJECT_NAME} stxxl)
构建时,我会遇到很多错误,因为要链接的库stxxl
与vcpkg
一起安装,具有以下代码段:
STXXL_BEGIN_NAMESPACE
template <class Type>
struct compat_unique_ptr {
#if __cplusplus >= 201103L && ((__GNUC__ * 10000 + __GNUC_MINOR__ * 100) >= 40400)
typedef std::unique_ptr<Type> result;
#else
// auto_ptr is inherently broken and is deprecated by unique_ptr in c++0x
typedef std::auto_ptr<Type> result;
#endif
};
STXXL_END_NAMESPACE
问题在于__cplusplus
的值为199711L
而不是正确的值201703L
,因此代码尝试使用在C ++ 17标准中已删除的auto_ptr
。 / p>
我还尝试在Visual Studio中手动设置该标志,并将此部分添加到CmakeLists.txt
if(MSVC)
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} /std:c++17 /Zc:__cplusplus")
endif()
但是没有任何变化。我还使用了一些C ++ 17功能,因此暂时不可以选择。
我可以手动更新stxxl
头文件以摆脱宏,但这是一种解决方法,每个开发人员都应创建相同的修复程序。相反,我想知道为什么会出现错误,以及如何使用Visual Studio和CMake将宏设置为正确的值。你有什么建议吗?
答案 0 :(得分:0)
我认为您需要的是target_compile_definitions,您可以在其中定义CMake中的任意宏并将其传递给编译器。
答案 1 :(得分:0)
据我从this得知,您需要手动添加/Zc:__cplusplus
(至少现在)。
您的CMakeList代码段对我有用,您确定__cplusplus
实际上设置不正确,而不仅仅是__GNUC__
宏,丢失了吗?
也就是说,我建议您测试一下VisualStudio版本是否真的足够新:
# /Zc:__cplusplus is required to make __cplusplus accurate
# /Zc:__cplusplus is available starting with Visual Studio 2017 version 15.7
# (according to https://docs.microsoft.com/en-us/cpp/build/reference/zc-cplusplus)
# That version is equivalent to _MSC_VER==1914
# (according to https://docs.microsoft.com/en-us/cpp/preprocessor/predefined-macros?view=vs-2019)
# CMake's ${MSVC_VERSION} is equivalent to _MSC_VER
# (according to https://cmake.org/cmake/help/latest/variable/MSVC_VERSION.html#variable:MSVC_VERSION)
if ((MSVC) AND (MSVC_VERSION GREATER_EQUAL 1914))
target_compile_options(stxxl INTERFACE "/Zc:__cplusplus")
endif()
注意:我使用target_compile_options向您的库目标界面添加了必要的选项。这样,所有依赖该库(通过target_link_libraries)的目标都将使用该标志进行编译,但是不需要(可能与之不兼容)的目标将无法获得该标志。
如果要将标志添加到自己的目标之一,则可以使用target_compile_options(stxxl PUBLIC "/Zc:__cplusplus")
(PUBLIC
代替INTERFACE
)。 INTERFACE
表示在其他目标依赖该目标时使用该选项,PRIVATE
表示该选项本身用于目标,而PUBLIC
表示两者。 (恕我直言this artical解释得很好。)
除此之外,您不应该通过编译标志设置c ++标准版本,而是如果需要更新版本,则使用target_compile_features并使用
set_target_properties(your_target_name PROPERTIES CXX_STANDARD 17)
(如果您只想使用较新的版本)。
在您的情况下,后者可能就足够了,因为如果c ++版本较旧,您的代码段将具有备用功能。
如果可以选择更改stxxl的代码(通过拉取请求等),则该代码可以测试_MSVC_LANG
和_MSC_VER
。这样一来,无需/Zc:__cplusplus
就可以工作。