Skip to content

CMake 基础用法

创建 CMake 项目大概需要以下几步:

  1. 首先需要一个目录当作工作空间,也叫项目目录
  2. 在这个目录中有一些 c/cpp 文件需要进行编译和链接
  3. 此时,需要一个 CMakeLists.txt 文件来进行项目管理。

创建标准 C 程序

这里以标准 C 编写一个 demo,如何将 cmake 参数传递到代码中,编写 main.c 文件,内容为:

c
#include <stdio.h>

int main (int argc, char *argv[])
{
    (void)(argc);
    (void)(argv);
    // VERSION 宏使用示例,可使用‘cmake -DVERSION=1.1.1’来指定不同的值
#ifdef VERSION
    printf("test version: %s\n", VERSION);
#else
    printf("test version: 1.0.0\n");
#endif  // DEBUG
    return 0;
}

然后在当前目录编写 CMakeLists.txt,内容为:

cmake
cmake_minimum_required(VERSION 3.13)

# If do't define version number, specify the version number
set(VERSION "1.0.0" CACHE STRING "define project version")

# project 有两种写法,这里建议使用这种写法,补全信息有助于后续使用
project(cdemo
    LANGUAGES C
    HOMEPAGE_URL https://github.com/Groveer/cdemo
    DESCRIPTION "c program demo."
    VERSION ${VERSION}
)

# 定义GNU标准安装目录,使用此定义可兼容不同发行版之间的安装目录差异
include(GNUInstallDirs)
# 设置C标准
set(CMAKE_C_STANDARD 11)
# 设置为检查C标准打开,若未设置CMAKE_C_STANDARD,则会报错
set(CMAKE_C_STANDARD_REQUIRED on)
# 设置包含当前目录,建议头文件的包含写全路径
set(CMAKE_INCLUDE_CURRENT_DIR ON)
# 打开所有编译警告,理论上项目中不允许有编译警告
# 关于编译选项,项目中不应该添加任何额外的参数,若发行版需要,则应由发行版的配置文件添加,若Debug需要,则应只添加到Debug模式中
# 在外部添加编译参数的方法,例:cmake -DCMAKE_C_FLAGS="-Wl,--as-needed"
set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -Wall -Wextra")
# 生成编译命令,用于支持clangd
set(CMAKE_EXPORT_COMPILE_COMMANDS ON)

# 若未指定安装前缀,则指定前缀为/usr
# 注意大部分发行版默认会将前缀设置为/usr/local,所以若需要安装到/usr,还需在编译时指定
if(CMAKE_INSTALL_PREFIX_INITIALIZED_TO_DEFAULT)
    set(CMAKE_INSTALL_PREFIX /usr)
endif()

# 若未指定编译类型,则指定为Release,防止某些发行版不指定编译类型
if (NOT CMAKE_BUILD_TYPE)
    set(CMAKE_BUILD_TYPE Release CACHE STRING "Choose Debug or Release" FORCE)
endif()

# 若目标库安装了.cmake文件,则可以直接使用find_package
#find_package(PkgConfig REQUIRED)
# 若目标库未安装.cmake文件,但是安装了.pc文件,则可以使用pkgconfig
#pkg_check_modules(PolkitQt REQUIRED IMPORTED_TARGET polkit-qt5-1)

# 建议对不同的模块进行分类,可使用通配符,可指定具名文件
file(GLOB_RECURSE SRCS
    "main.c"
)

set(BIN_NAME ${PROJECT_NAME})

# 生成可执行文件
add_executable(${BIN_NAME}
    ${SRCS}
)

# 这里针对当前二进制定义了一个宏,保存着版本号,方便程序代码中调用,若程序中需要自定义宏变量并且在代码中使用,可参考此方法。
# 注意CMAKE_PROJECT_XXX系列变量,某些属性是cmake帮助提供,但某些属性是project方法设置的,所以这里需要注意变量是否可用。
# 至于是使用add_compile_xxx还是target_compile_xxx,视情况而定,一个是指定所有对象,一个是指定具体的二进制
# target_compile_definitions(${BIN_NAME} PRIVATE
#     VERSION="${CMAKE_PROJECT_VERSION}"
# )

target_include_directories(${BIN_NAME} PUBLIC
    #PkgConfig::PolkitQt 
)

target_link_libraries(${BIN_NAME} PRIVATE
    #PkgConfig::PolkitQt 
)

# 指定安装目录,一般常用有3个: TARGETS(编译出来的目标二进制)、FILES(指定路径的文件,通常是配置文件或服务文件)、DIRECTORY(一般是配置文件较多时使用)。
install(TARGETS ${BIN_NAME} DESTINATION ${CMAKE_INSTALL_BINDIR})

创建基于 Qt 的 C++ 程序

与上一章做同样功能,但是使用基于 Qt 框架来开发,同样输出自定义版本号。编写 main.cpp 文件,内容为:

cpp
#include <QDebug>
#include <qglobal.h>

int main(int argc, char *argv[])
{
    Q_UNUSED(argc);
    Q_UNUSED(argv);
    // VERSION 宏使用示例,可使用‘cmake -DVERSION=1.1.1’来指定不同的值
    QString version;
#ifdef VERSION
    version = VERSION;
#else
    version = "1.0.1";
#endif  // DEBUG
    qDebug() << "test version: " << version << "!";
    return 0;
}

然后在当前目录编写 CMakeLists.txt,内容为:

cmake
cmake_minimum_required(VERSION 3.13)

# If do't define version number, specify the version number
set(VERSION "1.0.0" CACHE STRING "define project version")

# project 有两种写法,这里建议使用这种写法,补全信息有助于后续使用
project(qtdemo
    LANGUAGES CXX
    HOMEPAGE_URL https://github.com/Groveer/qtdemo
    DESCRIPTION "qt program demo."
    VERSION ${VERSION}
)

# 定义GNU标准安装目录,使用此定义可兼容不同发行版之间的安装目录差异
include(GNUInstallDirs)
# 设置C++标准
set(CMAKE_CXX_STANDARD 17)
# 设置为检查C++标准打开,若未设置CMAKE_CXX_STANDARD,则会报错
set(CMAKE_CXX_STANDARD_REQUIRED on)
# 若使用标准C++开发,则不需要以下两行,使用Qt则需要
set(CMAKE_AUTOMOC ON)
set(CMAKE_AUTORCC ON)
# 设置包含当前目录,建议头文件的包含写全路径
set(CMAKE_INCLUDE_CURRENT_DIR ON)
# 打开所有编译警告,理论上项目中不允许有编译警告
# 在外部添加编译参数的方法,例:cmake -DCMAKE_CXX_FLAGS="-Wl,--as-needed"
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -Wall -Wextra")
# 生成编译命令,用于支持clangd
set(CMAKE_EXPORT_COMPILE_COMMANDS ON)

# 若未指定安装前缀,则指定前缀为/usr
# 注意大部分发行版默认会将前缀设置为/usr/local,所以若需要安装到/usr,还需在编译时指定
if(CMAKE_INSTALL_PREFIX_INITIALIZED_TO_DEFAULT)
    set(CMAKE_INSTALL_PREFIX /usr)
endif()

# 若未指定编译类型,则指定为Release,防止某些发行版不指定编译类型
if(NOT CMAKE_BUILD_TYPE)
    set(CMAKE_BUILD_TYPE Release CACHE STRING "Choose Debug or Release" FORCE)
endif()

# 若目标库安装了.cmake文件,则可以直接使用find_package
#find_package(PkgConfig REQUIRED)
# 若目标库未安装.cmake文件,但是安装了.pc文件,则可以使用pkgconfig
#pkg_check_modules(PolkitQt REQUIRED IMPORTED_TARGET polkit-qt5-1)
find_package(QT NAMES Qt6 Qt5 REQUIRED COMPONENTS Core)
find_package(Qt${QT_VERSION_MAJOR} REQUIRED COMPONENTS Core Gui)

# 建议对不同的模块进行分类,可使用通配符,可指定具名文件
file(GLOB_RECURSE SRCS
    "main.cpp"
)

set(BIN_NAME ${PROJECT_NAME})

# 生成可执行文件
add_executable(${BIN_NAME}
    ${SRCS}
)

# 这里针对当前二进制定义了一个宏,保存着版本号,方便程序代码中调用,若程序中需要自定义宏变量并且在代码中使用,可参考此方法。
# 注意CMAKE_PROJECT_XXX系列变量,某些属性是cmake帮助提供,但某些属性是project方法设置的,所以这里需要注意变量是否可用。
# 至于是使用add_compile_xxx还是target_compile_xxx,视情况而定,一个是指定所有对象,一个是指定具体的二进制
# target_compile_definitions(${BIN_NAME} PRIVATE
#     VERSION="${CMAKE_PROJECT_VERSION}"
# )

# Qt 从5.15版本开始,可以直接使用Qt::Core,而不需要加版本号,但为了兼容性,把版本号加上为好
target_include_directories(${BIN_NAME} PUBLIC
    Qt${QT_VERSION_MAJOR}::Core
    #PkgConfig::PolkitQt 
)

target_link_libraries(${BIN_NAME} PRIVATE
    Qt${QT_VERSION_MAJOR}::Core
    #PkgConfig::PolkitQt 
)

# 指定安装目录,一般常用有3个: TARGETS(编译出来的目标二进制)、FILES(指定路径的文件,通常是配置文件或服务文件)、DIRECTORY(一般是配置文件较多时使用)。
install(TARGETS ${BIN_NAME} DESTINATION ${CMAKE_INSTALL_BINDIR})

博客内容遵循 CC BY-NC-SA 4.0 协议。