Makefile与CMake
CMake与Makefile详解
概述
CMake是一种跨平台编译工具,比make更为高级,使用起来要方便得多。CMake主要是编写CMakeLists.txt文件,然后用cmake命令将CMakeLists.txt文件转化为make所需要的makefile文件,最后用make命令编译源码生成可执行程序或共享库(so, shared object)。它的作用和Qt的qmake是相似的。
C/C++程序员肯定离不开Makefile和Cmake,因为如果对这两个工具不熟悉,那么你就不是一个合格的C/C++程序员。本文对Makefile和Cmake及它们的使用进行了详细的介绍。
一、Makefile详解
Makefile描述了整个工程的编译、连接等规则,makefile定义了一系列规则来指定:哪些文件需要编译以及如何编译、需要创建哪些库文件以及如何创建这些库文件、如何产生我们想要的可执行文件。使用Makefile,整个工程都可以完全自动化编译。而且Makefile可以有效的减少编译和连接的程序,只编译和连接那些修改的文件。
1.1 Makefile语法
Makefile包含了五个重要的东西:显示规则、隐晦规则、变量定义、文件指示和注释。
-
显示规则:说明了如何生成一个或多个目标。由Makefile指出要生成的文件和文件依赖的文件。
-
隐晦规则:基于Makefile的自动推导功能。
-
变量的定义:一般是字符串。
- 文件指示:
- 在Makefile中引用另外一个makefile文件;
- 根据某些规则指定Makefile中有效的部分;
- 多行。
- 注释:使用#表示注释。
Makefile有三个非常重要的变量:
$@
:目标文件$^
:所有依赖文件$<
:第一个依赖文件.PHONY
:伪目标文件
Makefile的执行过程如下:
- 在当前目录下寻找Makefile或makefile。
- 找到第一个文件中的第一个目标文件,和目标文件依赖的.o文件。
- 如果.o文件不存在,或是后面.o文件比target文件更新,那么它就会执行后面的语句来生成这个文件。
- 最后makefile会根据.o文件依赖的.h和.c文件生成.o文件。
注意事项:
- clean不要放在target前面。
-rm edit $(objects)
可忽略某些文件的问题。- Makefile中的命令,必须以Tab键分割。文件之间最好使用空格分割。
- 使用
-I
或--include-dir
参数,make会在这些目录下寻找头文件。 -L
指定库目录,相当于load lib dir,-lfb303
相当于链接libfb303.so。
g++编译常用选项:
-g
:调试信息-Wall
:显示所有警告-O1~3
:优化级别-lpthread
:多线程支持-j8
:多线程编译-D
:宏定义,例如-D_YUQIANG,那么#ifdef _YUQIANG为真。
1.2 Makefile示例
CC = gcc
RM = rm
CFLAGS += -D _YUQIANG
TARGETS := myapp
all:$(TARGETS)
$(TARGETS):main.c
$(CC) $(CFLAGS) $^ -o $@
clean:
-$(RM) -f *.o
-$(RM) -f $(TARGETS)
二、CMake详解
CMake是一个跨平台的安装(编译)工具,可以用简单的语句描述所有平台的安装(编译过程)。它能输出各种各样的makefile或者project文件,能测试编译器所支持的C++特性,类似UNIX下的automake。
2.1 CMake语法
常用CMake命令:
PROJECT(project_name)
:设置项目名称INCLUDE_DIRECTORIES(include)
:设置头文件路径SET(TEST_DIR ${DIR_SRCS})
:设置环境变量SET(LIBRARIES libm.so)
:设置外部库ADD_EXECUTABLE(../bin/bin ${TEST_DIR})
:设置可执行文件路径TARGET_LINK_LIBRARIES(../bin/bin ${LIBRARIES})
:设置链接库ADD_SUBDIRECTORY
:添加子目录
2.2 CMake示例
# project name
PROJECT(test_math)
# head file path
INCLUDE_DIRECTORIES(
include
)
# source directory
AUX_SOURCE_DIRECTORY(src DIR_SRCS)
# set environment variable
SET(TEST_MATH
${DIR_SRCS}
)
# set extern libraries
SET(LIBRARIES
libm.so
)
# add executable file
ADD_EXECUTABLE(${PROJECT_NAME} ${TEST_MATH})
# add link library
TARGET_LINK_LIBRARIES(${PROJECT_NAME} m)
三、CMake编译原理
3.1 CMake编译步骤
CMake编译基本分为两个步骤:
- cmake:指向CMakeLists.txt所在目录,例如:
mkdir build
cd build
cmake ..
这会生成编译所需的中间文件及makefile。
- make:根据生成的makefile编译程序:
make
3.2 实际项目示例
我们编写一个关于开平方的C/C++程序项目,即b = sqrt(a),以此理解整个CMake编译的过程。
项目结构:
.
├── build
├── CMakeLists.txt
├── include
│ └── b.h
└── src
├── b.c
└── main.c
CMakeLists.txt内容:
# cmake verson,指定cmake版本
cmake_minimum_required(VERSION 3.2)
# project name,指定项目的名称,一般和项目的文件夹名称对应
PROJECT(test_sqrt)
# head file path,头文件目录
INCLUDE_DIRECTORIES(
include
)
# source directory,源文件目录
AUX_SOURCE_DIRECTORY(src DIR_SRCS)
# set environment variable,设置环境变量,编译用到的源文件全部都要放到这里
SET(TEST_MATH
${DIR_SRCS}
)
# add executable file,添加要编译的可执行文件
ADD_EXECUTABLE(${PROJECT_NAME} ${TEST_MATH})
# add link library,添加可执行文件所需要的库
TARGET_LINK_LIBRARIES(${PROJECT_NAME} m)
编译和运行:
mkdir build
cd build
cmake ..
make
./test_sqrt
四、工具链关系说明
gcc是GNU Compiler Collection(GNU编译器套件),可以编译多种编程语言(C、C++、Objective-C、Fortran、Java等)
- 当程序只有一个源文件时,可以直接用gcc命令编译
- 当程序包含多个源文件时,使用make工具来管理编译过程
- make工具是一个智能的批处理工具,通过调用makefile中的命令进行编译和链接
- makefile类似于乐谱,make工具根据makefile中的命令调用编译器
- CMake可以更简单地生成makefile文件,并能跨平台生成对应平台的makefile
- CMake根据CMakeLists.txt文件生成makefile