使用CMake使用/ MT而不是/ MD进行编译


70

我在带有Windows SDK和NMake Makefile的Windows上使用CMake。

默认情况下,它使用/MD编译器开关进行编译。

我如何更改它以使用/MT开关进行编译?

Answers:


88

您可以修改CMAKE_CXX_FLAGS_<Build Type>和/或CMAKE_C_FLAGS_<Build Type>变量:

set(CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE} /MT")
set(CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} /MTd")

如果您的CMake标志已经包含/MD,则可以确保在插入点之后执行上述命令/MD(以后的添加将/MT覆盖冲突的现有选项),或者可以从头开始设置标志:

set(CMAKE_CXX_FLAGS_RELEASE "/MT")
set(CMAKE_CXX_FLAGS_DEBUG "/MTd")

或者,您可以通过执行以下操作分别用和替换现有的/MD/MDd值:/MT/MTd

set(CompilerFlags
        CMAKE_CXX_FLAGS
        CMAKE_CXX_FLAGS_DEBUG
        CMAKE_CXX_FLAGS_RELEASE
        CMAKE_C_FLAGS
        CMAKE_C_FLAGS_DEBUG
        CMAKE_C_FLAGS_RELEASE
        )
foreach(CompilerFlag ${CompilerFlags})
  string(REPLACE "/MD" "/MT" ${CompilerFlag} "${${CompilerFlag}}")
endforeach()

@Josh我刚刚更新了答案。如果仍然不完全是您想要的,请显示CMakeLists.txt的相关内容,我敢肯定会有一个简洁的答案。
Fraser

@Josh我在string(REPLACE...)命令中犯了一个小错误-现在已修复。
Fraser

29
这个请求很常见-奇怪的是CMake仍然没有宏或设置来支持此功能。我最终在几乎所有项目中都做到了这一点-特别是那些具有外部依赖项(例如GoogleMock)的项目,它们对编译器ABI拥有自己的默认意见
kert 2013年

@kert-不仅奇怪,而且很神秘。
杰里米(Jeremy)2016年

谁能告诉我应该在vcpkg的哪个文件中进行此更改?
Gunnar

55

CMake最终在3.15版中使用MSVC_RUNTIME_LIBRARYtarget属性添加了对此的适当支持:

cmake_minimum_required(VERSION 3.15)
cmake_policy(SET CMP0091 NEW)
project(my_project)

add_executable(foo foo.c)
set_property(TARGET foo PROPERTY
             MSVC_RUNTIME_LIBRARY "MultiThreaded$<$<CONFIG:Debug>:Debug>")

您也可以通过设置CMAKE_MSVC_RUNTIME_LIBRARY变量来指定全局默认值。


1
cmake_policy(SET CMP0091 NEW)在3.16.3上不起作用,确实需要吗?
弗拉基米尔·加马良

4
@VladimirGamalyan与策略一样,如果您cmake_minimum_required足够新,则无需显式请求新行为。我主要在回答中包括了该策略,以便人们知道该功能存在一个策略,如果他们关心兼容性问题,则可以查找它。
ComicSansMS

1
至少对于现代cmake,这应该是公认的解决方案。这比黑客CMAKE_CXX_FLAGS更加不透明...
大卫·卡拉

@DavidKarla除了这个问题是在提出原始问题六年后给出的,并不是说被问的人自2015
。– John Cvelth

6

看来,对于Visual Studio的2017年15CMake的3.12的方式来代替/MD/MT是加入该片段到的CMakeLists.txt文件:

if(MSVC)
    add_compile_options(
        $<$<CONFIG:>:/MT> #---------|
        $<$<CONFIG:Debug>:/MTd> #---|-- Statically link the runtime libraries
        $<$<CONFIG:Release>:/MT> #--|
    )
endif()

我在官方CMake存储库中找到了此解决方案:https : //gitlab.kitware.com/cmake/cmake/issues/18390


注意:此块应添加在add_executableadd_library
Bernardo Ramos

2

我必须使用它set( ... CACHE ... FORCE)来覆盖MSVC的默认缓存。

如果我不使用此方法,则MSVC仍会输出/MD选项。

set(CompilerFlags
        CMAKE_CXX_FLAGS
        CMAKE_CXX_FLAGS_DEBUG
        CMAKE_CXX_FLAGS_RELEASE
        CMAKE_CXX_FLAGS_MINSIZEREL
        CMAKE_CXX_FLAGS_RELWITHDEBINFO
        CMAKE_C_FLAGS
        CMAKE_C_FLAGS_DEBUG
        CMAKE_C_FLAGS_RELEASE
        CMAKE_C_FLAGS_MINSIZEREL
        CMAKE_C_FLAGS_RELWITHDEBINFO
        )
foreach(CompilerFlag ${CompilerFlags})
    string(REPLACE "/MD" "/MT" ${CompilerFlag} "${${CompilerFlag}}")
    set(${CompilerFlag} "${${CompilerFlag}}" CACHE STRING "msvc compiler flags" FORCE)
    message("MSVC flags: ${CompilerFlag}:${${CompilerFlag}}")
endforeach()

经过数小时的浪费时间,这是唯一对我有用的方法。谢谢!
错误的

尽管这似乎仍然可以动态链接一些标准库:MSVCP140.dll,VCRUNTIME140_1.dll,VCRUNTIME140.dll,api-ms-win-crt-string-l1-1-0.dll ...
错误

@Miscreant我唯一能建议的就是检查cmake缓存中所有生成的命令行参数。cmake的规则很复杂,而MSVC的系统却很杂乱。
肖恩

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.