cmakelists.txt中set(cmake_cxx_standard 17)在windows上不生效,因msvc忽略该变量,需配合set(cmake_cxx_standard_required on)和set(cmake_cxx_extensions off)才能强制标准模式并禁用扩展。

为什么 CMakeLists.txt 里写 set(CMAKE_CXX_STANDARD 17) 在 Windows 上不生效?
因为 visual studio 工具链默认忽略该变量,它只影响 Makefile 或 Ninja 生成器下的编译器参数。MSVC 的 c++ 标准由 set(CMAKE_CXX_STANDARD_REQUIRED ON) 和 set(CMAKE_CXX_EXTENSIONS OFF) 配合才能真正约束。
- 必须加上
set(CMAKE_CXX_STANDARD_REQUIRED ON),否则 MSVC 可能回退到 C++14 -
set(CMAKE_CXX_EXTENSIONS OFF)强制使用标准模式(禁用 MSVC 扩展),否则std::string_view等可能意外可用但移植到 GCC/Clang 时失败 - 在
project()之后立即设置,顺序错位会导致部分 target 不继承
如何让同一个 CMakeLists.txt 同时适配 GCC、Clang 和 MSVC 的警告选项?
不能硬编码 -Wall -Wextra,MSVC 不认这些 flag;也不能只写 /W4,GCC 会报错。得用条件判断 + target_compile_options() 按编译器分支设置。
- 用
$<strequal></strequal>判断CMAKE_CXX_COMPILER_ID:值可能是gnu、Clang、MSVC - 对 GCC/Clang 推荐加
-Wno-unused-parameter -Wno-missing-field-initializers,避免第三方头文件触发误报 - MSVC 要补
/permissive-(启用标准一致性)和/utf-8(源码编码统一),否则含中文字符串字面量时 linux/macos 编译失败
find_package(OpenSSL REQUIRED) 在 macOS 上总失败,但 Homebrew 明明装了 OpenSSL?
macOS 的 find_package() 默认只查系统路径(/usr),而 Homebrew 把 OpenSSL 装在 /opt/homebrew(Apple Silicon)或 /usr/local(Intel)。CMake 不自动搜这些位置。
- 运行 CMake 时显式传
-DOpenSSL_ROOT_DIR=/opt/homebrew/opt/openssl(M1/M2)或-DOpenSSL_ROOT_DIR=/usr/local/opt/openssl(Intel) - 更稳妥的做法是在
CMakeLists.txt中加set(CMAKE_PREFIX_PATH "${CMAKE_PREFIX_PATH} /opt/homebrew /usr/local") - 避免用
brew link openssl—— 它会污染系统/usr/local,导致 Linux 容器内构建时 header 版本错乱
交叉编译 android 时,add_executable() 生成的可执行文件为什么在手机上直接报 not executable: 64-bit ELF file?
因为没指定目标架构和 ABI,CMake 默认用主机工具链。Android NDK 提供的独立工具链必须通过 CMAKE_TOOLCHAIN_FILE 指定,且需匹配 ABI(如 arm64-v8a)和 API 级别。
立即学习“C++免费学习笔记(深入)”;
- 调用 CMake 时必须带:
-DCMAKE_TOOLCHAIN_FILE=$NDK/build/cmake/android.toolchain.cmake -DANDROID_ABI=arm64-v8a -DANDROID_PLATFORM=android-21 -
add_executable()的二进制默认是 host 可执行格式,Android 上只能跑add_library(... SHARED)或用add_executable(... IMPORTED)包装已编译的 so - NDK r21+ 默认禁用 exceptions 和 RTTI,若代码用了
try/catch,得额外加-DANDROID_CPP_FEATURES="exceptions rtti"
跨平台真正的难点不在语法,而在每个平台对“标准”的实现偏差——比如 MSVC 对模板推导的宽容度、macOS 对符号可见性的默认处理、Android 对 C++ runtime 的静态链接要求。漏掉任意一个,编译可能成功,但运行时崩得毫无征兆。