CMake快速入门教程
前言
Make工具因遵循不同的规范和标准,执行的Makefile的格式也是不同。主流的Make工具包括:
- GNU Make
- QT的qmake
- 微软的 MS nmake
- BSD的 pmake
每个平台都有自己的工具,则带来了很大的平台兼容性问题。CMake是一种跨平台的编译工具。
准备阶段:
- 安装cmake
- 编写CMake配置文件CMakeLists.txt
基本流程:
- 执行
cmake PATH或者ccmake PATH命令,将CMakeLists.tx文件转化为所需要的Makefile文件。其中PATH为CMAKLISTs.txt所在的目录 - 执行make命令,编译原码生成可执行程序,或者库文件
单目录,单文件
一个简单的样例:
# CMake的最低版本要求
cmake_minimum_required (VERIONS 2.8)
# 项目信息
project(Demo)
# 指定生成目标
add_executable(Demo demo.cc)
语法规则:
- 命令、空格、注释组成
- 命令是不区分大小写的
- 符号
#后面内容为注释 - 参数之间使用
空格分隔
单目录,多文件
在实际项目中,一般不会只有一个demo.cc源码文件,常为一个目录下多个源文件。假设目录结构如下:
./Demo
|-- main.cc
|-- foo.cc
|-- foo.h
此时的CMakeLists.txt内容可以更新为如下:
# CMake的最低版本要求
cmake_minimum_required (VERIONS 2.8)
# 项目信息
project(Demo)
# 指定生成目标
add_executable(Demo demo.cc foo.cc)
即只需要在add_executable里把依赖的foo.cc源文件添加进来即可。
但引入另一个问题:新增的源文件越来越多,总不能一个个手动加进来吧?
cmake中有 aux_source_directory命令,会查找指定目录下所有源文件,并存放到指定的变量名中:
# CMake的最低版本要求
cmake_minimum_required (VERIONS 2.8)
# 项目信息
project(Demo)
# 查找当前目录下所有源文件
aux_source_directory(. DIR_SRCS)
# 指定生成目标
add_executable(Demo ${DIR_SRCS})
多目录,多文件
针对一个项目中包含了多了层级目录,且每个目录下都包含一些源文件。若目录结构如下:
./Demo
|-- main.cc
|-- utils
|-- foo.cc
|-- foo.h
我们需要分别在Demo和utils目录下各自编写一个CMakeLists.txt文件。
为了方便,可以先将utils目录里的文件编译成静态库,再由main函数调用。
根目录中的CMakeLists.txt
# CMake的最低版本要求,如果不满足则报错
cmake_minimum_required (VERIONS 2.8 FATAL_ERROR)
# 项目信息
project(Demo)
# 添加 math 子目录
add_subdirectory(utils)
# 指定生成目标
add_executable(Demo main.cc)
# 添加链接库
target_link_libraries(Demo utils)
add_subdirectory表示会处理子目录下的CMakeLists.txt和源代码target_link_libraries表示main执行文件需要链接一个名为utils的链接库
utils目录中的CMakeLists.txt
# 查找当前目录下的所有源文件、并保存到 DIR_LIB_SRCS 变量中
aux_source_directory(. DIR_LIB_SRCS)
# 生成链接库
add_library(utils ${DIR_LIB_SRCS})
add_library会将所有源文件编译为静态链接库
其他编译选项
如下是一个项目的CMakeLists.txt:
cmake_minimum_required(VERSION 2.8)
project(Demo)
# 定一个开关选项,支持cmake时通过 -DUSE_MYUTILS=OFF 指定
option(USE_MYUITLS "whether use customized math" ON)
# 自定义分支逻辑
if(USE_MYUITLS)
include_directories("{PROJECT_SOURCE_DIR}/utils")
add_subdirectory(utils)
set(EXTRA_LIBS ${EXTRA_LIBS} utils)
endif(USE_MYUITLS)
# 查找目录下所有源文件
aux_source_directory(. DIR_SRCS)
# 添加执行文件
add_executable(Demo ${DIR_SRCS})
# 链接静态库
target_link_libraries(Demo ${EXTRA_LIBS})
关于option的生效机制,这里详细解释下。如main.cc中的代码:
#include <stdio.h>
#include <stdlib.h>
#include "config.h" // 此头文件是cmake自动生成的
#ifdef USE_PYUTILS
#include "utils/foo.h"
#else
#include <foo.h> // 假设标准库有foo.h头文件
#endif
为了打通CMakeLists.txt一键便携式配置,我们需要编写一个config.h.in文件:
#cmakedefine USE_MYUTILS
这样,在执行cmake命令时,就可以根据配置的参数,自动生成option相关的头文件。若指定-DUSE_MYUTILS=ON时,config.h中的内容为:
#define USE_MYUTILS
若为OFF时,则config.h的内容为:
/* #undef USE_MYUTILS */
安装和测试
camke支持安装和测试,通过在生成Makefile后,使用make install和make test来执行。
接上述样例,首先在utils/CMakeLists.txt中加上如下内容:
# 指定utils库的安装路径
install(TARGETS utils DESTINATION bin)
install(FILES utils.h DESTINATION include)
在Demo/CMakeLists.txt添加如下内容:
# 指定安装路径
install(TARGETS Demo DESTINATION bin)
install(FILES "${PROJECT_BINARY}/config.h" DESTINATION include)
原理 & 流程:
- cmake编译产出的
Demo文件和库libUtils.o文件将会被复制到/usr/local/bin中 - 头文件
utils.h和config.h则会被赋值到/use/local/include中 - 可以通过
CMKAE_INSTALL_PREFIX修改默认安装的根目录/usr/local
关于测试,CMake提供一个称为CTest的测试工具,通过add_test命令添加:
# 启用测试
enable_testing()
# 测试程序是否成功运行, arg*为函数接收的参数
add_test(test_run Demo arg1 arg2)
add_test(test_usage Demo)
set_tests_properties(test_usage PROPERTIES PASS_REGULAR_EXPRESSION "Usage: .....")
add_test(test_result Demo 10 2)
# 测试输出的结果是否包含字符串 "is 100"
set_tests_properties(test_result PROPERTIES PASS_REGULAR_EXPRESSION "is 100")
支持gdb
CMake支持gdb的方式很简单,只需指定Debug模式下开启-g,一个简单的样例如下:
set(CMAKE_BUILD_TYPE "Debug")
# debug模式下编译选项
set(CMAKE_CXX_FLAGS_DEBUG "$ENV{CXXFLAGS} -O0 -Wall -g -ggdb")
# release模式下编译选项
set(CMAKE_CXX_FLAGS_RELEASE "$ENV{CXXFLAGS} -O3 -Wall")
小结
CMake的语法主要以命令、空格、参数来组成
可以通过
set(<variable> <value>)设置变量的值if语法
if(<condition>)
<commands>
elseif(<condition>) # optional block, can be repeated
<commands>
else() # optional block
<commands>
endif()
for语法
# usage 1:
foreach(<loop_var> <items>)
<commands>
endforeach() # usage 2:
foreach(<loop_var> RANGE <stop>)
while语法
while(<condition>)
<commands>
endwhile()
附录:
CMake快速入门教程的更多相关文章
- CMake快速入门教程-实战
http://www.ibm.com/developerworks/cn/linux/l-cn-cmake/ http://blog.csdn.net/dbzhang800/article/detai ...
- 转:CMake快速入门教程-实战
CMake快速入门教程:实战 收藏人:londonKu 2012-05-07 | 阅:10128 转:34 | 来源 | 分享 0. 前言一个多月 ...
- CMake快速入门教程:实战
转自http://blog.csdn.net/ljt20061908/article/details/11736713 0. 前言 一个多月前,由于工程项目的需要,匆匆的学习了一下cmake的使 ...
- [转]CMake快速入门教程:实战
转自http://blog.csdn.net/ljt20061908/article/details/11736713 0. 前言 一个多月前,由于工程项目的需要,匆匆的学习了一下cmake的使 ...
- 专为设计师而写的GitHub快速入门教程
专为设计师而写的GitHub快速入门教程 来源: 伯乐在线 作者:Kevin Li 原文出处: Kevin Li 在互联网行业工作的想必都多多少少听说过GitHub的大名,除了是最大的开源项目 ...
- EntityFramework6 快速入门教程
EntityFramework6 快速入门教程 不得不说EF在国内实在是太小众,相关的技术文章真实屈指可数,而且很多文章都很旧了,里面使用的版本跟如今的EF6差别还是比较大.我刚开始弄这个的时候真是绕 ...
- Apple Watch开发快速入门教程
Apple Watch开发快速入门教程 试读下载地址:http://pan.baidu.com/s/1eQ8JdR0 介绍:苹果为Watch提供全新的开发框架WatchKit.本教程是国内第一本A ...
- 指示灯组与3个复位按钮的介绍Arduino Yun快速入门教程
指示灯组与3个复位按钮的介绍Arduino Yun快速入门教程 1.4.2 指示灯组 指示灯组的放大图如图1.5所示. 图1.5 指示灯组 各个指示灯对应的功能如下: q RX:对应于0号端口, ...
- 游戏控制杆OUYA游戏开发快速入门教程
游戏控制杆OUYA游戏开发快速入门教程 1.2.2 游戏控制杆 游戏控制杆各个角度的视图,如图1-4所示,它的硬件规格是本文选自OUYA游戏开发快速入门教程大学霸: 图1-4 游戏控制杆各个角度的 ...
- Query 快速入门教程
Query 快速入门教程 http://www.365mini.com/page/jquery-quickstart.htm#what_is_jquery jquery常用方法及使用示例汇总 http ...
随机推荐
- Chrome浏览器安装离线插件Markdown Here
前言 近期刚开通了微信公众号,想在上面发表一些自己的文章,由于排版问题很浪费时间,公众号后台本身不支持Markdown语法,所以就需要利用Markdown Here插件使得公众号后台支持Markdow ...
- 从优秀到卓越:成为DevOps专家的7项软技能
在我的职业生涯中,遇见过许多专业人士,他们在技术上非常健全,对自己的领域和技术有很好的掌握和专业知识,但是由于缺乏软技能,他们错过了晋升.现场机会.高级技术面试以及职业生涯中的机会.很震惊吧,技术好却 ...
- 巴延兴:从主导多个SIG组到OpenHarmony“代码贡献之星”,我是如何做到的?
编者按:在 OpenHarmony 生态发展过程中,涌现了大批优秀的代码贡献者,本专题旨在表彰贡献.分享经验,文中内容来自嘉宾访谈,不代表 OpenHarmony 工作委员会观点. 巴延兴 深圳开鸿数 ...
- C#-GroupBox包含控件,如何获取这些控件的名称
您可以使用 Enumerable.OfType在GroupBox中查找和投射您的RadioButtons: var radioButtons = groupBox1.Controls.OfType&l ...
- apache 服务器配置常用知识点合集
前言 因为当年周围同学都在学php,最为简单的就是学php 好就业啊,写个一些php,最后放弃了,apache也看了两眼吧.下面是我使用有记录的,没有记录的我后面会补上. 正文 域名配置 1.取消 N ...
- flask售后评分系统
做软件行业的公司,一般都有专业的售前售后团队,还有客服团队,客服处理用户反馈的问题,会形成工单,然后工单会有一大堆工单流程,涉及工单的内部人员,可能会有赔付啥的,当然,这是有专业的售前.售后.客服团队 ...
- mybatis generator生成mapper接口后的代理类,很方便使用。
1.spring 配置: <bean id="superMapperProxy" class="com.qws.v1.daoImpl.MapperProxy&quo ...
- Node 中的 Process 理解,有哪些常用方法?
一.是什么 process 对象是一个全局变量,提供了有关当前 Node.js进程的信息并对其进行控制,作为一个全局变量 我们都知道,进程计算机系统进行资源分配和调度的基本单位,是操作系统结构的基础, ...
- JS中通过url动态获取图片大小的方法小结(两种方法)
很多时候再项目中,我们往往需要先获取图片的大小再加载图片,但是某些特定场景,如用过cocos2d-js的人都知道,在它那里只能按比例缩放大小,是无法设置指定大小的图片的,这就是cocos2d-js 的 ...
- 力扣49(java)-字母异位词分组(中等)
题目: 给你一个字符串数组,请你将 字母异位词 组合在一起.可以按任意顺序返回结果列表. 字母异位词 是由重新排列源单词的字母得到的一个新单词,所有源单词中的字母通常恰好只用一次. 示例 1: 输入: ...