• CMake:构建时运行自定义命令add_custom_command


    CMake:构建时运行自定义命令add_custom_command

    导言

    项目的构建目标取决于命令的结果,这些命令只能在构建系统生成完成后的构建执行。CMake提供了三个选项来在构建时执行自定义命令:

    • 使用add_custom_command编译目标,生成输出文件。
    • add_custom_target的执行没有输出。
    • 构建目标前后,add_custom_command的执行可以没有输出。

    这三个选项强制执行特定的语义,并且不可互换。接下来的我们将分别学习具体的用法。

    项目结构

    本项目比较简单,我们通过对add_custom_command的简单使用,来探索它的功能。

    .
    ├── CMakeLists.txt
    ├── config
    │   └── config.txt
    └── test.cpp
    
    • 1
    • 2
    • 3
    • 4
    • 5

    项目地址:

    https://gitee.com/jiangli01/tutorials/tree/master/cmake-tutorial/chapter5/02

    CMakeLists.txt

    cmake_minimum_required(VERSION 3.0)
    
    project(custom_command_example)
    
    # Set static library to lib file
    set(CMAKE_ARCHIVE_OUTPUT_DIRECTORY ${CMAKE_BINARY_DIR}/lib)
    # Set dynamic library to lib file
    set(CMAKE_LIBRARY_OUTPUT_DIRECTORY ${CMAKE_BINARY_DIR}/lib)
    # Set dynamic runtime library or exetuable file to bin file
    set(CMAKE_RUNTIME_OUTPUT_DIRECTORY ${CMAKE_BINARY_DIR}/bin)
    
    add_executable(${PROJECT_NAME} test.cpp)
    
    if (MSVC)
      file(GLOB config_file "${CMAKE_SOURCE_DIR}/config/*.txt")
      add_custom_command(TARGET ${PROJECT_NAME} POST_BUILD
                         COMMAND ${CMAKE_COMMAND} -E copy_if_different
                         ${config_file}
                         $<TARGET_FILE_DIR:${PROJECT_NAME}>)
    elseif(UNIX)
      file(GLOB config_file "${CMAKE_SOURCE_DIR}/config/*.txt")
      file(COPY ${config_file} DESTINATION ${CMAKE_RUNTIME_OUTPUT_DIRECTORY})
    endif()
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23

    add_custom_commandCMake 中的一个命令,用于向构建系统添加自定义的构建规则或自定义命令。这可以用于执行各种任务,如生成源文件、拷贝文件、执行脚本等。它允许在 CMake 构建过程中定义一些额外的操作。

    以下是 add_custom_command 命令的一般语法和参数:

    add_custom_command(
        OUTPUT output1 [output2...]
        COMMAND command1 [ARGS] [command2 [ARGS] ...]
        [MAIN_DEPENDENCY depend]
        [DEPENDS depend [depend ...]]
        [WORKING_DIRECTORY dir]
        [COMMENT comment]
        [VERBATIM]
        [APPEND]
        [USES_TERMINAL]
    )
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • OUTPUT output1 [output2...]: 指定命令执行后生成的输出文件。这些文件通常是构建过程的目标,可以是可执行文件、库文件、数据文件等。

    • COMMAND command1 [ARGS] [command2 [ARGS] ...]: 定义要执行的命令。这可以是外部命令、脚本或自定义操作。

    • MAIN_DEPENDENCY depend: 指定主要的依赖项,通常是影响命令执行的文件。如果 depend 被修改,命令将重新运行。

    • DEPENDS depend [depend ...]: 指定其他依赖项。这些文件会触发命令重新运行,如果它们被修改。

    • WORKING_DIRECTORY dir: 指定命令执行的工作目录。

    • COMMENT comment: 可选,用于描述自定义命令的文本注释。

    • VERBATIM: 可选,告诉 CMake 保持命令的参数不变,不进行任何转义。

    • APPEND: 可选,将新的自定义命令追加到同一输出文件上。

    • USES_TERMINAL: 可选,指示命令是否使用终端。

    通常,add_custom_command 用于在构建期间执行一些非标准的操作,例如生成代码、转换文件格式、运行测试或其他自定义任务。这可以帮助您在 CMake 构建系统中添加额外的步骤,以满足项目的特定需求。

    本项目需要在Windows系统中构建,以探究add_custom_command 命令的功能。该命令在本项目中的主要功能时将根目录下config·文件夹下的所有txt本文文件拷贝到可执行文件所在的目录。

    之所有在Windows中的命令和Linux中的命令不一样,根本原因是由于操作系统的文件系统的不同造成的。

    test.cpp

    #include 
    #include 
    #include 
    
    int main() {
      // 文件路径
      std::string file_name = "config.txt";
    
      // 打开文件
      std::ifstream file(file_name);
    
      // 检查文件是否成功打开
      if (!file.is_open()) {
        std::cerr << "无法打开文件: " << file_name << std::endl;
        return 1;
      }
    
      std::string line;
    
      // 逐行读取文件内容
      while (std::getline(file, line)) {
        std::cout << line << std::endl;
      }
    
      // 关闭文件
      file.close();
    
      return 0;
    }
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30

    运行结果

    version: 0.0.1
    author: jiangli
    email: 1316762810@qq.com
    
    • 1
    • 2
    • 3

    最后,祝大家变得更强!!!

  • 相关阅读:
    C++类中的常成员和静态成员
    医保移动支付和接口对接开发
    java毕业设计新闻稿件管理系统Mybatis+系统+数据库+调试部署
    【阿里国际笔试】编程1&3
    开发者专访|我从「人间地狱」的算法岗内卷中,倔强地踏出了一条路 前路漫漫,星芒万里
    【mysql】2006 - Server has gone away
    GBase 8d的特性-可扩展性
    低代码技术这么香,如何把它的开发特点发挥到极致?
    LeetCode - 547 省份数量
    边际图和组合折线图
  • 原文地址:https://blog.csdn.net/jjjstephen/article/details/134003158