C++20 源代码信息类std::source_location 实现简易日志打印功能
C++20 使用源代码信息类std::source_location 实现简易日志打印功能
·
传统方法获取源代码信息
传统方法是使用预定义宏来获取,比如有__LINE__
、__FUNCTION__
和__FILE__
。
比如一个简易的日志系统,我们的log方法用于打印日志信息:
void log(PSTR level, PSTR file, PSTR func, int line, PSTR fmt, ...){
char str[256] = {0};
va_list arglst;
va_start(arglst,fmt);
vsnprintf(str,sizeof(str), fmt, arglst);
printf("[%s][%s] %s in function %s() at line:%d : %s\n",__TIME__, level, file, func, line, str);
va_end(arglst);
}
每次都要传入__FILE__
、__FUNCTION__
和__LINE__
非常麻烦,如果你使用默认的参数的话,因为其实宏,在预编译阶段就会解析为函数声明的行。而且如果你使用可变参数的话,就无法使用默认参数了。
所以我们可以通过声明宏函数来解决,解析时直接解析为调用行。然后我们简易的日志类实现如下:
#include <stdio.h>
#include <stdarg.h>
using PSTR = const char*;
#define DEBUG_LOG(...) log("DEBUG", __FILE__, __FUNCTION__, __LINE__, __VA_ARGS__)
#define WARNING_LOG(...) log("WARN", __FILE__, __FUNCTION__, __LINE__, __VA_ARGS__)
#define ERROR_LOG(...) log("ERROR", __FILE__, __FUNCTION__, __LINE__, __VA_ARGS__)
void log(PSTR level, PSTR file, PSTR func, int line, PSTR fmt, ...){
char str[256] = {0};
va_list arglst;
va_start(arglst,fmt);
vsnprintf(str,sizeof(str), fmt, arglst);
printf("[%s][%s] %s in function %s() at line:%d : %s\n",__TIME__, level, file, func, line, str);
va_end(arglst);
}
int main(){
int a,b;
printf("Type your division expression:");
scanf("%d/%d",&a,&b);
if(b!=0)
printf("Result: %f",a/b);
else
WARNING_LOG("The expression \'%d/%d\' is not right, cause you cannot divide by 0.",a,b);
return 0;
}
样例1:
Type your division expression:2/7
Result: 0.000000
样例2:
Type your division expression:8/0
[00:01:00][WARN] D:\SingleSources\CPP\3.cpp in function main() at line:26 : The expression '8/0' is not right, cause you cannot divide by 0.
std::source_location
source_location 类于C++20被引入,它表示关于源码的具体信息,它的结构以及使用都非常简单。
成员函数
我们使用其构造函数和一个静态成员函数current()
创建实例。此外还有访问域的函数:
line()
:获取行号。column()
:获取列号。file_name()
:获取文件名。function_name()
:获取函数域名。
实例
废话不多说,直接上实例。
#include <iostream>
#include <string>
#include <source_location>
void log(std::string level,
std::string message,
const std::source_location& location = std::source_location::current()){
std::cout << level
<< location.file_name() << " in function "
<< location.function_name() << " at "
<< location.line() << ':' << location.column() << ": "
<< message << '\n';
}
int main()
{
log("[DEBUG]","Test");
}
输出内容:
[DEBUG]2.cpp in function int main() at 17:8: Test
开放原子开发者工作坊旨在鼓励更多人参与开源活动,与志同道合的开发者们相互交流开发经验、分享开发心得、获取前沿技术趋势。工作坊有多种形式的开发者活动,如meetup、训练营等,主打技术交流,干货满满,真诚地邀请各位开发者共同参与!
更多推荐
已为社区贡献1条内容
所有评论(0)