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 ...) 正则表达式中子表达式的用法。
posted @ 2022-05-10 11:41  larybird  阅读(898)  评论(0编辑  收藏  举报