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 实现:
- CMakeLists.txt 中获取时间;
- 将时间相关变量写入 http://config.h.in 文件中;
- 使用 configure_file() 将 http://config.h.in 中的变量转换为 C 中可识别的宏定义,存入 config.h 文件中;
- 源码中使用宏定义获取对应时间。
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 编写http://config.h.in 文件
将上述变量写入 http://config.h.in 文件中,然后使用 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 中增加以下语句,同时将 http://config.h.in 中的 @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 中。因此,在 http://config.h.in 中,需要将 @TIME_DAY@ 改为 @TIME_DAY_NUM@。
cmake 中 string 的更多功能可查看官方文档:https://cmake.org/cmake/help/latest/command/string.html#string
执行结果:
5. 总结
这篇文章中包含了 3 个知识点:
- CMakeLists.txt 中获取系统时间;
- configure_file() 用法;
- string(REGEX REPLACE ...) 正则表达式中子表达式的用法。
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 分享4款.NET开源、免费、实用的商城系统
· 全程不用写代码,我用AI程序员写了一个飞机大战
· MongoDB 8.0这个新功能碉堡了,比商业数据库还牛
· 白话解读 Dapr 1.15:你的「微服务管家」又秀新绝活了
· 记一次.NET内存居高不下排查解决与启示