• CLion同一目录下多个main函数


    转载自:https://zhuanlan.zhihu.com/p/157646534

    ============================================

    使用CLion 刷题解决多个main函数问题的终极方法

    在写 C++ 的题目的时候经常会遇到这样的问题,写了多个 cpp 文件,在 clion 中编译报错不能同时存在多 main 函数。

    下面列举几种方法:

    方法1:重定义Main

    在每个文件中通过重定义的方法来解决,在写某道算法时,对main进行重定义,

     

     

     

     

    运行完后再修改回去,这样就能接下去就能再重定义为main函数接着运行了。

     

     

    优点:不需要修改配置文件

    缺点:会让源码文件中多出一些奇奇怪怪的代码,降低代码可阅读性!

    方法2:手动修改CmakeList.txt

    通过手动添加add_executable(编译文件名 源码文件地址)

    cmake_minimum_required(VERSION 3.16)
    project(C_AND_C_Plus_Plus_Practise)
    
    set(CMAKE_CXX_STANDARD 14)
    
    add_executable(MAIN main.cpp)
    
    add_executable(CHelloWord ./C_Practise/HelloWord.c)
    add_executable(C3test ./C_Practise/3_变量/test.c)
    add_executable(DataStructureLove DataStructure/1Introduction/Love.cpp)
    add_executable(DataStructureSqList DataStructure/2LinearList/SqList.cpp)
    add_executable(DataStructureSqList1 DataStructure/2LinearList/SqList1.cpp)
    add_executable(DataStructureLinkList DataStructure/2LinearList/LinkList.cpp)
    add_executable(DataStructureLinkList1 DataStructure/2LinearList/LinkList1.cpp)

    优点:只修改配置文件,不会影响源码的可读性

    缺点:每新建一个文件,就得修改配置文件,较为繁琐!

    方法3:在CMake文件中编写自动生成程序

    在Cmake文件中编写程序,自动生成编译后的文件名!

    基础版:

    # 遍历项目根目录下所有的 .cpp 文件
    file (GLOB files *.cpp)
    foreach (file ${files})
    string(REGEX REPLACE ".+/(.+)\\..*" "\\1" exe ${file})
    add_executable (${exe} ${file})
    message (\ \ \ \ --\ src/${exe}.cpp\ will\ be\ compiled\ to\ bin/${exe})
    endforeach ()

    上边儿这段代码,只会遍历根目录下的cpp文件,不会遍历根目录下的二级目录。

    进阶版:

    # 遍历项目根目录及二级目录下所有的 .cpp 文件
    file (GLOB files *.cpp */*cpp)
    foreach (file ${files})
    string(REGEX REPLACE ".+/(.+)\\..*" "\\1" exe ${file})
    add_executable (${exe} ${file})
    message (\ \ \ \ --\ src/${exe}.cpp\ will\ be\ compiled\ to\ bin/${exe})
    endforeach ()

    进阶版就是手动添加访问二级目录的规则,同样要访问三级目录就是再添加一个*/*/*.cpp

    最优版:

    官方文档提供一个解决方法GLOB_RECURSE,它会自动遍历工程文件根目录下的所有文件目录。

    # 遍历项目根目录下所有的 .cpp 文件
    file (GLOB_RECURSE files *.cpp)
    foreach (file ${files})
    string(REGEX REPLACE ".+/(.+)\\..*" "\\1" exe ${file})
    add_executable (${exe} ${file})
    message (\ \ \ \ --\ src/${exe}.cpp\ will\ be\ compiled\ to\ bin/${exe})
    endforeach ()

    优点:方便省时

    缺点:这种方法要求所有cpp文件命名不重复,不能含有中文,不能含有‘/’等字符!因为它就是直接Copy你的源码文件名的。

     

     

     

     

     

     

     

     

     

    参考:https://zhuanlan.zhihu.com/p/442889385

    ===========================================

    cmake:string(REGEX REPLACE ...)

    1. 需求

    项目要求,下位机传给上位机的版本号为数字格式,并且其中要包含软件发布时的日期(年份最低两位)。比如,软件版本号为 4,发布日期为 2021 年 3 月 2 日。那么传给上位机的数据为 “1a 03 02 04”

    2. 方案

    可以基于 cmake 中的 configure_file 实现:

    1. CMakeLists.txt 中获取时间;
    2. 将时间相关变量写入  文件中;
    3. 使用 configure_file() 将  中的变量转换为 C 中可识别的宏定义,存入 config.h 文件中;
    4. 源码中使用宏定义获取对应时间。

    3. 实现

    3.1 获取系统时间

    cmake 中使用 string(TIMESTAMP <output_variable> [<format_string>] [UTC]) 获取系统时间。比如:

    cmake_minimum_required(VERSION 3.20)
    project(cmake-string)
    
    #获取年月日时分秒
    string(TIMESTAMP COMPILE_TIME %Y%m%d%H%M%S)
    #获取年份后两位,使用 %Y 获取完整年份
    string(TIMESTAMP TIME_YEAR %y)
    #获取月份
    string(TIMESTAMP TIME_MONTH %m)
    #获取日期
    string(TIMESTAMP TIME_DAY %d)
    message(STATUS "compile time:${COMPILE_TIME}")
    configure_file (
      "${CMAKE_CURRENT_LIST_DIR}/config.h.in"
      "${CMAKE_CURRENT_BINARY_DIR}/config.h"
      )
    include_directories(${CMAKE_CURRENT_BINARY_DIR})
    add_executable(${PROJECT_NAME} main.c)

    3.2 编写 文件

    将上述变量写入  文件中,然后使用 configure_file() 将其转为 config.h 文件后,即可再 .c 文件中使用。

    #ifndef VERSION_CONFIG_H
    #define VERSION_CONFIG_H
    #cmakedefine TIME_YEAR @TIME_YEAR@
    #cmakedefine TIME_MONTH @TIME_MONTH@
    #cmakedefine TIME_DAY @TIME_DAY@
    #endif

    3.3 C 源码

    #include <stdio.h>
    #include <stdlib.h>
    #include <string.h>
    #include "config.h"
    int main(int argc, char *argv[])
    {
        printf("Build time:%d-%d-%d\n", TIME_YEAR, TIME_MONTH, TIME_DAY);
    return 0;
    }

    3.4 执行结果

    Easy?NO!


    4. 问题

    4.1 出现问题

    今天,2021 年 12 月 8 日,我要发布软件。

    编译:

    config.h:

    #ifndef VERSION_CONFIG_H
    #define VERSION_CONFIG_H
    #define TIME_YEAR 21
    #define TIME_MONTH 12
    #define TIME_DAY 08
    #endif

    问题就出在 #define TIME_DAY 08 上面。

    C 语言中,以 '0' 开头的数值表示八进制数,八进制数的有效值为 0~7。所以,上面的宏定义肯定是错误的。

    如何解决?

    4.2 解决问题

    我想到的办法是在 CMakeLists.txt 中将变量 TIME_DAY 的值前面的 ‘0’ 去掉。这里,我使用 string(REGEX REPLACE ...) 语句达到此目的。

    在 CMakeLists.txt 中增加以下语句,同时将  中的 @TIME_DAY@ 改为 @TIME_DAY_NUM@

    string(REGEX REPLACE "(^[0])([1-9]*)" "\\2" TIME_DAY_NUM ${TIME_DAY})

    这语句的意思是:如果变量 TIME_DAY 的值以 '0' 开头,那么就将 '0' 去掉,只保留 '0' 以后的数值,并将数值保存在变量 TIME_DAY_NUM 中。

    "(^[0])([1-9]*)" 和 "\\2" 说明:

    上述正则表达式使用了子表达式。子表达式使用小括号——'()' 进行分隔,使用时用 '\1'、'\2'... 进行引用。上述语句中,'(^[0])' 为子表达式 1,'([1-9]*)' 为子表达式 2。'"\\2"' 表示只保留子表达式 2 的内容(这种用法我在 source insight 中也用过)。

    比如,TIME_DAY 的值为 '08','(^[0])' 会匹配 '0', '([1-9]*)' 匹配 '8','"\\2"' 会将 '8' 保存到变量 TIME_DAY_NUM 中。因此,在  中,需要将 @TIME_DAY@ 改为 @TIME_DAY_NUM@

    cmake 中 string 的更多功能可查看官方文档:

    执行结果:

    5. 总结

    这篇文章中包含了 3 个知识点:

      1. CMakeLists.txt 中获取系统时间;
      2. configure_file() 用法;
      3. string(REGEX REPLACE ...) 正则表达式中子表达式的用法。
  • 相关阅读:
    使用集合组织相关数据
    引用类型传递 ListView展示数据
    关于ArrayList线程安全解决方案
    Java异常错误的面试题及答案
    希尔排序
    Struts2的简单认识
    新闻发布系统
    了解vo pojo javabean dto
    Spring MVC 笔记及简单了解
    jsp的九大内置对象
  • 原文地址:https://www.cnblogs.com/hd92/p/16252842.html
Copyright © 2020-2023  润新知