【问题标题】:Cannot set __cplusplus to C++17 standard with Visual Studio and CMake无法使用 Visual Studio 和 CMake 将 __cplusplus 设置为 C++17 标准
【发布时间】:2019-11-27 20:14:49
【问题描述】:

我有一个使用 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)

构建时出现很多错误,因为我链接的库stxxlvcpkg 一起安装,具有以下代码:

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

我也尝试在 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 将宏设置为正确的值。你有什么建议吗?

【问题讨论】:

  • Visual Studio 不支持所有 C++17 功能。 docs.microsoft.com/en-us/cpp/overview/…
  • @JesperJuhl 我做到了,我已经尝试设置/Zc 标志,正如我在帖子中所说的那样,但没有任何改变。也许我需要以另一种方式设置视觉开关?
  • 请注意,无论如何您都必须破解标头,因为即使您让 MSVC 设置正确的 __cplusplus,您也不太可能将其设置为 __GNUC__
  • 是的,我已经注意到了,我正在 vcpkg 存储库中创建一个拉取请求,但 __cplusplus 问题仍然存在,这是一个普遍问题,与库没有严格关系。

标签: c++ visual-studio cmake c++17 visual-studio-2019


【解决方案1】:

据我从this 得知,您需要手动添加/Zc:__cplusplus(至少现在是这样)。

你的 CMakeList sn-p 对我有用,你确定 __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++ 版本较旧,您的 coden-p 有一个回退。


如果可以选择更改 stxxl 的代码(通过拉取请求等),则该代码可以测试 _MSVC_LANG_MSC_VER。这样它就可以在不需要/Zc:__cplusplus 的情况下工作。

【讨论】:

    【解决方案2】:

    我认为你需要的是target_compile_definitions,你可以在CMake中定义任意宏并传递给编译器。

    【讨论】:

      猜你喜欢
      • 2021-10-28
      • 2022-11-09
      • 2021-03-01
      • 2020-02-09
      • 2021-08-29
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      相关资源
      最近更新 更多