make
用来编译c++项目,make
命令根据Makefile
中配置的编译链接关系;由于Makefile
文件的制作是个大工程,因此出现了cmake工具
,cmake
根据CMakeLists.txt
来执行cmake命令;
使用CMake编写跨平台工程的流程如下:
(1)编写源文件
(2)编写CMakeLists.txt
(3)由CMake根据CMakeLists.txt来生成相应的makefile文件
(4)使用make并根据makefile调用gcc来生成相应的可执行文件
CMake是一个可以跨平台的编译工具,可以用简单的语句来描述所有平台的编译过程。他能够输出各种各样的 makefile 或者工程文件。和make与makefile类似,我们在使用CMake时同样也需要一个文件来提供规则,这个文件就是CMakeLists
编写CMakeLists.txt最常用的功能就是调用其他的.h头文件和.so/.a库文件,将.cpp/.c/.cc文件编译成可执行文件或者新的库文件。
# 指定cmake最小版本
cmake_minimun_version(VERSION 3.4.1)
# 本CMakeLists.txt的project名称
# 会自动创建两个变量,PROJECT_SOURCE_DIR和PROJECT_NAME
# ${PROJECT_SOURCE_DIR}:本CMakeLists.txt所在的文件夹路径
# ${PROJECT_NAME}:本CMakeLists.txt的project名称
# CMAKE_BINART_DIR, PROJECT_BINARY_DIR, _BINARY_DIR:这三个变量的含义一样。
project(xxx)
# 获取路径下所有的.cpp/.c/.cc文件,并赋值给变量中
aux_source_directory(路径 变量)
# 给文件名/路径名或其他字符串起别名,用${变量}获取变量内容
set(变量 文件名/路径/...)
# 添加编译选项
add_definitions(编译选项)
# 打印消息
message(消息)
# 编译子文件夹的CMakeLists.txt
add_subdirectory(子文件夹名称)
# 将.cpp/.c/.cc文件生成可执行文件
add_executable(可执行文件名称 文件)
# 将.cpp/.c/.cc文件生成.a静态库 // add_library(库文件名称 STATIC 文件)
# 注意,库文件名称通常为libxxx.so,在这里只要写xxx即可
# 生成静态库(默认)
add_library(myCommonLib STATIC util.cpp)
# 生成动态库
add_library(myCommonLib SHARED util.cpp)
# 规定.h头文件路径
include_directories(路径)
# 规定.so/.a库文件路径
link_directories(路径)
# 对add_library或add_executable生成的文件进行链接操作
# 注意,库文件名称通常为libxxx.so,在这里只要写xxx即可
target_link_libraries(库文件名称/可执行文件名称 链接的库文件名称)
# 如果是链接的库,则运行的时候需要把它拷贝到可执行文件名称同目录下
如何设置DGB调试模式请看这里
add_library(demo demo.cpp test.cpp util.cpp)
# 发现一个目录下所有的源代码文件,并将列表存储再一个变量中(不会递归遍历目录)
aux_source_directory(. SRC_LIST)
add_library(demo ${SRC_LIST})
file(GLOB SRC_LIST "*.cpp" "protocol/*.cpp")
add_library(demo ${SRC_LIST})
#或者
file(GLOB SRC_LIST "*.cpp")
file(GLOB SRC_PROTOCOL_LIST "protocol/*.cpp")
add_library(demo ${SRC_LIST} ${SRC_PROTOCOL_LIST})
#或者
aux_source_directory(. SRC_LIST)
aux_source_directory(protocol SRC_PROTOCOL_LIST)
add_library(demo ${SRC_LIST) ${SRC_PROTOCOL_LIST)
add_definitions
的功能和C/C++中的#define
是一样的
# 添加一个宏定义,设置后,代码中即可使用 `TEST_DEBUG`这个宏,等同在代码中增加 #define TEST_DEBUG
add_definitions(-DTEST_DEBUG)
于是cpp代码中,就可以使用TEST_DEBUG
这个宏了
#ifdef TEST_DEBUG
...
#else
...
#endif
问题是,由谁来驱动定义这个宏呢?通过结合options指令可以实现
# 设置一个宏选项默认值ON,
# option选项 不会影响到cpp代码(不会添加#define宏)
option(TEST_DEBUG "test" ON)
if(TEST_DEBUG )
message("itis" ${TEST_IT_CMAKE})
add_definitions(-DTEST_DEBUG )
endif()
最后,在执行cmake时,就可以设置该选项了,并传递该值下去;
cmake .. -DTEST_DEBUG=1
通过cmake-gui也可以很方便的配置选项,如下:
add_definitions("-Wall -g")
#没加之前
gcc -c main.c -o test
#添加之后,相当于
gcc -g -Wall -c main.c -o tes
#在指定目录下搜索一个库, 保存在变量MY_LIB中
find_library(MY_LIB "usr/lib/libmylib.so" )
#链接这个库
target_link_libraries(${PROJECT_NAME} ${MY_LIB})
变量 | 说明 |
---|---|
PROJECT_SOURCE_DIR | 工程根目录 |
PROJECT_BINARY_DIR | 运行cmake命令的目录,通常是${PROJECT_SOURCE_DIR}/build |
PROJECT_NAME | 返回通过project命令定义的项目名称 |
CMAKE_CURRENT_SOURCE_DIR | 当前处理的CMakeLists.txt所在的路径 |
CMAKE_CURRENT_BINARY_DIR | target 编译目录 |
CMAKE_CURRENT_LIST_DIR | CMakeLists.txt的完整路径 |
CMAKE_CURRENT_LIST_LINE | 当前所在的行 |
CMAKE_MODULE_PATH | 定义自己cmake模块所在的路径。SET(CMAKE_MODULE_PATH ${PROJECT_SOURCE_DIR}/cmake),然后可以用INCLUDE命令来调用自己的模块 |
EXECUTABLE_OUTPUT_PATH | 重新定义目标二进制可执行文件的存放位置 |
LIBRARY_OUTPUT_PATH | 重新定义目标链接库的存放位置 |
$ENV{name}
set(ENV{name} value)
创建如下目录结构
- /myCpp
- CMakeLists.txt
- /build
- /src
- myHello.cpp
myHello.cpp
#include
int main() {
std::cout<<"Hello world!"<<std::endl;
return 0;
}
CMakeLists.txt
cmake_minimum_required(VERSION 3.0)
project(Hello)
MESSAGE(STATUS"This is BINARY_dir" ${PROJECT_BINARY_DIR})
MESSAGE(STATUS"This is SOURCE_dir"${PROJECT_SOURCE_DIR})
aux_source_directory(./src DIR_SRCS)
add_executable(${PROJECT_NAME} ${DIR_SRCS})
编译过程
$ cd build
$ cmake ..
-- The C compiler identification is GNU 5.4.0
-- The CXX compiler identification is GNU 5.4.0
-- Check for working C compiler: /usr/bin/cc
-- Check for working C compiler: /usr/bin/cc -- works
-- Detecting C compiler ABI info
-- Detecting C compiler ABI info - done
-- Detecting C compile features
-- Detecting C compile features - done
-- Check for working CXX compiler: /usr/bin/c++
-- Check for working CXX compiler: /usr/bin/c++ -- works
-- Detecting CXX compiler ABI info
-- Detecting CXX compiler ABI info - done
-- Detecting CXX compile features
-- Detecting CXX compile features - done
STATUS"This is BINARY_dir"/home/bing/bingCpp/myDemo/build
STATUS"This is SOURCE_dir"/home/bing/bingCpp/myDemo
-- Configuring done
-- Generating done
-- Build files have been written to: /home/bing/bingCpp/myDemo/build
$ make
Scanning dependencies of target Hello
[ 50%] Building CXX object CMakeFiles/Hello.dir/src/main.cpp.o
[100%] Linking CXX executable Hello
[100%] Built target Hello