项目日志——相关技术补充(1)
文章目录
- 不定参数函数
- 宏函数
- C语言
- C++
不定参数函数
不定参数函数指的是参数个数不确定的情况,例如printf或者scanf
宏函数
正常调用printf
printf("[%s:%d]-%s\n",__FILE__ , __LINE__, "日志样例");
用...
表示参数不定,fmt对后面的参数进行替换
#define LOG1(fmt, ...)\printf("[%s:%d]-" fmt, __FILE__, __LINE__, ##__VA_ARGS__ );int main()
{LOG1("日志样例");return 0;
}
fmt表示传递的参数包,直接进行打印,__VA_ARGS__接收参数,##表示当只有一个参数时,删除前面的逗号,这样就不会报错了
C语言
#include <cstdarg>
void LOG2(int cnt, ...)
{va_list ap;va_start(ap, cnt); // 获取第一个参数的起始地址for (int i = 0; i < cnt; i++){int num = va_arg(ap, int); // 获取之后的参数,需要指定类型std::cout << num << ' ';}std::cout << std::endl;va_end(ap); // ap指针置空
}int main()
{// LOG1("日志样例");LOG2(2, 1, 2);return 0;
}
需要头文件
ap表示头指针,va_start(ap, cnt)表示跳过第一个表示数字的参数,得到开头
va_arg(ap, int)返回参数并跳过,需要指定参数类型
具体参见man手册
但是当我们无法确定参数类型时,这样的直接调用就无法使用了
因此我们要实现一个类似于printf的函数
void LOG3(const char *fmt, ...)
{va_list ap;va_start(ap, fmt);char *res;int ret = vasprintf(&res, fmt, ap);if (ret != -1)printf("%s", res);free(res);va_end(ap);
}int main()
{// LOG1("日志样例");// LOG2(2, 1, 2);LOG3("[%s:%d]-%s", __FILE__, __LINE__, "日志样例");return 0;
}
这里的vasprintf是一个字符串处理函数,可以将fmt的格式化字符串解析,并将ap中的参数放入,再放到res中,返回值是表示放入的参数的个数,-1表示失败
C++
void LOG4()
{std::cout << std::endl;
}template <typename T, typename... Args>
void LOG4(const T &v, Args &&...args)
{std::cout << v;if (sizeof ... (args)){LOG4(std::forward<Args>(args)...);}
}int main()
{// LOG1("日志样例");// LOG2(2, 1, 2);// LOG3("[%s:%d]-%s", __FILE__, __LINE__, "日志样例");LOG4("日志样例1", "日志样例2", "日志样例3");return 0;
}
这里主要用到的是模板函数,用的是递归的思想
定义了一个模板参数包类型,然后用右值引用给接住,因为我们不想更改原先参数的各类性质
sizeof的功能在C++11中扩展了,可以用来获取参数包中的参数数目
forward是一个完美转发,不更改右值的属性,(args)…表示解包,也就是将其转换为一个参数加一个参数包的形式
最后完全解包传入的是一个空值,因此需要对模板进行特化,写一个重载空函数