cmake入门

它是一个项目构建工具,开源,跨平台,并且简单易用。项目管理文件为CMakeLists.txt,生成时一般使用外部构建,在项目根目录下创建build目录,进入到build目录中执行cmake ..,然后make。入门篇不介绍脚本具体格式,只介绍最基础的使用。

1.最简单的可执行程序

项目包含:main.cpp。CMakeLists.txt:

CMAKE_MINIMUM_REQUIRED(VERSION 3.5)

PROJECT(hello)

ADD_EXECUTABLE(hello main.cpp)

2.包含头文件和cpp的可执行程序

项目包含:include/hello.h src/hello.cpp src/main.cpp。CMakeLists.txt:

CMAKE_MINIMUM_REQUIRED(VERSION 3.5)

PROJECT(hello)

SET(SOURCES
    src/hello.cpp
    src/main.cpp
)

ADD_EXECUTABLE(hello ${SOURCES})

TARGET_INCLUDE_DIRECTORIES(hello PRIVATE ${PROJECT_SOURCE_DIR}/include)

3.构建静态库

项目文件结构同2。CMakeLists.txt:

CMAKE_MINIMUM_REQUIRED(VERSION 3.5)

PROJECT(hello)

# 静态库hello_static

ADD_LIBRARY(hello_static STATIC src/hello.cpp)
TARGET_INCLUDE_DIRECTORIES(hello_static PUBLIC ${PROJECT_SOURCE_DIR}/include)

# 可执行程序
ADD_EXECUTABLE(hello src/main.cpp)

TARGET_LINK_LIBRARIES(hello PRIVATE hello_static)

4.构建动态库

项目文件结构同2。CMakeLists.txt:

CMAKE_MINIMUM_REQUIRED(VERSION 3.5)

PROJECT(hello)

# 动态库hello_shared
ADD_LIBRARY(hello_shared SHARED src/hello.cpp)
SET_TARGET_PROPERTIES(hello_shared PROPERTIES VERSION 1.2 SOVERSION 1) #动态库版本号设置
TARGET_INCLUDE_DIRECTORIES(hello_shared PUBLIC ${PROJECT_SOURCE_DIR}/include) 

# 可执行程序
ADD_EXECUTABLE(hello src/main.cpp)

TARGET_LINK_LIBRARIES(hello PRIVATE hello_shared)

5.安装

在2基础上增加配置文件test.conf。CMakeLists.txt:

CMAKE_MINIMUM_REQUIRED(VERSION 3.5)

PROJECT(hello)

# library
ADD_LIBRARY(hello_lib SHARED src/hello.cpp)

TARGET_INCLUDE_DIRECTORIES(hello_lib PUBLIC ${PROJECT_SOURCE_DIR}/include)

# exe
ADD_EXECUTABLE(hello src/main.cpp)

TARGET_LINK_LIBRARIES(hello PRIVATE hello_lib)

# install
INSTALL(TARGETS hello DESTINATION bin)

INSTALL(TARGETS hello_lib LIBRARY DESTINATION lib)

INSTALL(DIRECTORY ${PROJECT_SOURCE_DIR}/include/ DESTINATION include) #注意三个目标目录的写法是不同的

INSTALL(FILES test.conf DESTINATION etc)

可以使用make install DESTDIR=/XXX或者cmake -DCMAKE_INSTALL_PREFIX=/XXX ..来改变安装目录

6.指定编译类型

项目文件只有main.cpp。CMakeLists.txt:

CMAKE_MINIMUM_REQUIRED(VERSION 3.5)

IF(NOT CMAKE_BUILD_TYPE AND NOT CMAKE_CONFIGURATION_TYPES)
  MESSAGE("set build type to 'RelWithDebInfo' as none was specified.")
  SET(CMAKE_BUILD_TYPE RelWithDebInfo CACHE STRING "choose the type of build" FORCE)
  SET_PROPERTY(CACHE CMAKE_BUILD_TYPE PROPERTY STRINGS "Debug" "Release" "MinSizeRel" "RelWithDebInfo")
ENDIF()
PROJECT(hello)

ADD_EXECUTABLE(hello main.cpp)

使用时cmake .. -DCMAKE_BUILD_TYPE=Release

7.使用编译选项

项目文件只有main.cpp,但是它的内容使用了预处理标志。

 1 #include <iostream>
 2 
 3 int main(int argc, char *argv[])
 4 {
 5    std::cout << "Hello Compile Flags!" << std::endl;
 6 
 7    // only print if compile flag set
 8 #ifdef EX2
 9   std::cout << "Hello Compile Flag EX2!" << std::endl;
10 #endif
11 
12 #ifdef EX3
13   std::cout << "Hello Compile Flag EX3!" << std::endl;
14 #endif
15 
16    return 0;
17 }
CMAKE_MINIMUM_REQUIRED(VERSION 3.5)

SET(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -DEX2" CACHE STRING "set c++ compiler flags" FORCE)

PROJECT(hello)

ADD_EXECUTABLE(hello main.cpp)

TARGET_COMPILE_DEFINITIONS(hello PRIVATE EX3)

构建时可以使用cmake .. -DCMAKE_CXX_FLAG="-DEX3"

8.使用第三方库

项目只有main.cpp。CMakeLists.txt:

CMAKE_MINIMUM_REQUIRED(VERSION 3.5)

PROJECT(hello)

FIND_PACKAGE(Boost REQUIRED COMPONENTS filesystem system)

SET(CMAKE_ALLOW_LOOSE_LOOP_CONSTRUCTS ON) # if 标志

IF(Boost_FOUND)
  MESSAGE("boost found")
ELSE()
  MESSAGE(FATAL_ERROR "cannot find boost")
ENDIF()

ADD_EXECUTABLE(hello main.cpp)

TARGET_LINK_LIBRARIES(hello PRIVATE Boost::filesystem)

9.子项目管理

有三个子项目,每个子项目有各自目录和项目文件。总项目CMakeLists.txt:

CMAKE_MINIMUM_REQUIRED(VERSION 3.5)

PROJECT(hello)

ADD_SUBDIRECTORY(A)
ADD_SUBDIRECTORY(B)
ADD_SUBDIRECTORY(C)

 

posted @ 2020-11-25 21:36  soso101  阅读(133)  评论(0编辑  收藏  举报