前言
相信每个人都写过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
依赖于OpenCV
,main
又依赖于my_library
,那么main
就会间接依赖于OpenCV
。在这个例子中,my_library
和main
这两个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
声明头文件包含路径。
此外,每个命令都用到了PRIVATE
和PUBLIC
关键字。在CMake的官方说明中,称PRIVATE
声明的依赖为build-requirement,INTERFACE
声明的依赖为usage-requirement,PUBLIC
声明的依赖相当于同时声明了PRIVATE
和INTERFACE
。这里的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
网友评论