CMake最佳实践

前言

相信每个人都写过CMakeLists,然而,“一千个读者心中有一千个哈姆雷特”,一千个程序员也能写出一千种CMakeLists。这是因为CMake在发展的过程中始终保持向后兼容,在不断添加新特性的同时,仍然保留旧的语法规则。这样一来,同一个问题就会有多种写法。虽然无论哪种写法都可以成功构建,但在2019年的今天,我们应该与时俱进,摒弃不好的用法,采用官方推荐的最佳用法。这就是本文的主题。

面向Target编程

首先需要明确的是,CMake本身就是一种编程语言。我们所写的CMakeLists,其实就是在用CMake语法来编程,实现构建的功能。

我们习惯于这样写CMakeLists:

find_package(OpenCV REQUIRED) 

include_directories(${OpenCV_INCLUDE_DIRS})

add_library(my_library SHARED my_library.cpp)
target_link_libraries(my_library ${OpenCV_LIBRARIES})

add_executable(main main.cpp)
target_link_libraries(main my_library)

这种写法被无数人使用,但它存在严重的缺陷。请思考,如果我们构建的是一个库,当这个库被其它程序调用的时候,如何传递依赖?比如上面的例子,my_library依赖于OpenCVmain又依赖于my_library,那么main就会间接依赖于OpenCV。在这个例子中,my_librarymain这两个Target是放在一起创建的。但实际工程应用中,库和使用该库的程序应该是分开构建的,在构建main的过程中就势必需要获得它所有的间接依赖,否则在编译期可能找不到头文件,在链接期可能出现“未定义的引用”。

你可能会想,间接依赖的问题不应该由CMake自动帮我们完成吗。从CMake设计者的角度来考虑,如果他要实现这一功能,就必须把include_directories中的所有目录导出到间接依赖。但他不能这样做,因为显然大部分头文件都只在内部使用,作为API的头文件只是一小部分。所以不同用途的头文件必须使用不同的标识区分,之后才可以由CMake负责导出。

另一方面,如果同一个CMakeLists中包含了多个Target,单一的include_directories就显得不太合理,应该为每个Target单独设置。

Modern CMake

CMake从3.0开始进入Modern时代,也就是前文所说的面向Target编程。下面我们用一个具体的例子讲解如何做到这一点。

例子包含一个库MyLibrary和一个可执行程序App,但我们会在两个工程中分别构建它们。

首先来看MyLibrary库的目录结构:

my_library
-- cmake
   -- MyLibraryConfig.cmake
-- include
   -- my_library
      -- my_library.h
-- src
   -- my_library.cpp
-- CMakeLists.txt

头文件和源文件不必说了,直接看怎么写CMakeLists.txt。首先常规部分,声明工程名称,查找依赖库OpenCV

cmake_minimum_required(VERSION 3.5)
project(MyLibrary VERSION 1.0.0 LANGUAGES CXX)

find_package(OpenCV REQUIRED)

接下来创建Target。

## Add an empty library first. Then set properties for it.
add_library(MyLibrary)
target_compile_features(MyLibrary PRIVATE cxx_std_11)
target_sources(MyLibrary PRIVATE src/my_library.cpp)
target_include_directories(MyLibrary
        PUBLIC
            $<INSTALL_INTERFACE:include>
            $<BUILD_INTERFACE:${CMAKE_CURRENT_SOURCE_DIR}/include>
        PRIVATE
            ${OpenCV_INCLUDE_DIRS}
        )
target_link_libraries(MyLibrary PRIVATE ${OpenCV_LIBRARIES})

与传统CMake不同的是,我们用target_compile_features替代了对变量CMAKE_CXX_FLAGS的赋值。用target_sources声明源文件列表。用target_include_directories声明头文件包含路径。

此外,每个命令都用到了PRIVATEPUBLIC关键字。在CMake的官方说明中,称PRIVATE声明的依赖为build-requirement,INTERFACE声明的依赖为usage-requirement,PUBLIC声明的依赖相当于同时声明了PRIVATEINTERFACE。这里的build表明了该依赖仅存在于构建阶段,而usage则表明该依赖存在于这个库的使用阶段。举个简单的例子,如果我们的库依赖于OpenCV,但我们暴露给用户的接口与OpenCV毫无关系,那么这个依赖就是PRIVATE依赖。本文的案例就属于这种情况。

接下来,安装Target到系统目录中。

## We firstly install the generated libraries to /usr/local. The path
## comes from GNUInstallDirs, which includes lots of predefined system
## paths.
include(GNUInstallDirs)
install(TARGETS MyLibrary
        EXPORT MyLibraryTargets
        LIBRARY DESTINATION ${CMAKE_INSTALL_LIBDIR}
        ARCHIVE DESTINATION ${CMAKE_INSTALL_LIBDIR}
        )

## Then we install the auto-generated target file, in which have many
## exported names and paths of our target.
install(EXPORT MyLibraryTargets
        FILE MyLibraryTargets.cmake
        NAMESPACE MyLibrary::
        DESTINATION ${CMAKE_INSTALL_LIBDIR}/cmake/MyLibrary)

## And we should install the header files.
install(DIRECTORY include/
        DESTINATION ${CMAKE_INSTALL_INCLUDEDIR}
        )

## Finally, install the <package>Config.cmake file, which is provided
## for users.
install(FILES ${CMAKE_CURRENT_LIST_DIR}/cmake/MyLibraryConfig.cmake
        DESTINATION ${CMAKE_INSTALL_LIBDIR}/cmake/MyLibrary)

每一步都有详细的注释,大致流程是把生成的库文件拷到安装路径下,然后把生成的MyLibraryTargets.cmake文件拷到安装路径下,然后把头文件、MyLibraryConfig.cmake文件也拷到安装路径下。

需要特别指出的是,MyLibraryConfig.cmake文件是需要开发者自己写的,该文件的用途是让使用者通过find_packge找到这个库。好在这个文件并不难写,只有下面几行。

## Get the directory path of the <target>.cmake file
get_filename_component(MyLibrary_CMAKE_DIR "${CMAKE_CURRENT_LIST_FILE}" DIRECTORY)

## Add the dependencies of our library
include(CMakeFindDependencyMacro)
find_dependency(OpenCV REQUIRED)

## Import the targets
if(NOT TARGET MyLibrary::MyLibrary)
    include("${MyLibrary_CMAKE_DIR}/MyLibraryTargets.cmake")
endif()

里面总共做了两件事,第一是用find_dependency找到依赖库,这是我们作为库的作者所必须负责做的事情(因为用户根本不知道我们用到了OpenCV)。第二是导入MyLibraryTargets.cmake,这个文件里保存了前面我们声明的各种依赖的名称和路径。

现在,我们就可以编译、安装MyLibrary。接下来,看看怎么使用我们刚刚安装好的库。

App工程的目录结构如下:

app
-- main.cpp
-- CMakeLists.txt

CMakeLists.txt是这样写的:

cmake_minimum_required(VERSION 3.5)
project(App VERSION 1.0.0 LANGUAGES CXX)

find_package(MyLibrary REQUIRED)

## Create the executable target
add_executable(App main.cpp)
target_compile_features(App PRIVATE cxx_std_11)
target_link_libraries(App PRIVATE MyLibrary::MyLibrary)

可以说是非常清爽了,完全不必关心对于OpenCV的间接依赖。链接库的方式也从传统的${MyLibrary_LIBRARIES}变成了MyLibrary::MyLibrary

这个示例到这里就结束了,虽然非常简单,但已经给出了Modern CMake的大体框架。如果每个C++开发者都遵循Modern CMake的构建模式,整个C++开源社区将会变得更加高效。

完整代码可以从我的GitHub下载:jingedawang/modern_cmake_example

参考资料

Meeting C++ 2018: More Modern CMake Deniz Bahadir
C++ Now 2017: Effective CMake Daniel Pfeifer
It's Time To Do CMake Right Pablo
An Introduction to Modern CMake Henry Schreiner
CMake Documentation

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 158,847评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,208评论 1 292
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,587评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,942评论 0 205
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,332评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,587评论 1 218
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,853评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,568评论 0 198
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,273评论 1 242
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,542评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,033评论 1 260
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,373评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,031评论 3 236
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,073评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,830评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,628评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,537评论 2 269