如何重载采用可变数量参数的函数
How to overload a function taking variable number of arguments
我需要区分(重载)两个函数 - 一个接受单个 const char*
参数,另一个接受至少两个参数 - const char*
后跟一个或多个参数。
即基本上:
void f(const char *str);
void f(const char *format, ...)
我希望 f("hello")
调用第一个版本,f("hello %d", 10)
调用第二个版本。上面的重载不会工作,因为编译器发现 f("hello")
不明确。
所以我尝试了这个:
void f(const char *str);
template<typename T>
void f(const char *str, T tt, ...);
这使得重载解析工作正常。但我最终遇到了另一个问题。第二个函数应该转发 printf 样式用法的参数。所以我有类似的东西:
template <typename T>
void f ( const char *format, T tt, ... )
{
(T)tt;
va_list varlist;
va_start(varlist, format);
vprintf(format, varlist);
va_end(varlist);
}
现在第二个参数 tt
不再是可变参数列表的一部分,并且使用 format
does not seem to work.
调用 va_start()
有什么方法可以达到我想要的效果吗?
也许使用模板不是最好的主意,因为第二个错误似乎比第一个更难修复。函数的重载很好,如果你在一个单独的文件中尝试,你会发现它更有效。
#include <iostream>
bool f(const char* str) {
return true;
}
bool f(const char* str,int n) {
return false;
}
int main(int argc,char* argv[]) {
std::cout << f("Hi") << std::endl; //Gives 1
std::cout << f("Hi",10) << std::endl; //Gives 0
return 0;
}
您也可以尝试创建一个命名空间并将函数放入其中,从而确保没有其他文件共享可能导致该错误的函数名称。
如果您使用可变参数模板,您可以完成您想要的:
#include <iostream>
void f(const char* str)
{
std::cout << "single arg: " << str << "\n";
}
template <typename ...T>
void f(const char *format, T... args)
{
printf(format, args...);
}
int main()
{
f("hello");
f("formatted %d", 10);
}
- 问:为什么编译出错?
- 答:因为“void f(const char *format, ...)”的范围覆盖了“void f(const char *str);”。
所以可爱的解决方案是你最好使用 arg num 来区分 2 个函数。
像这样:
void func_with_one_arg(const char *str) {
printf("func_with_one_arg");
}
void f(int arg_num, const char *format, ...) {
if (arg_num==1) return func_with_one_arg(format);
printf("func_with_multi_arg\n");
va_list ap;
va_start(ap, format);
va_end(ap);
}
如果你想自动计算变量arg_num,使用这个宏来帮助计算。
#define VA_LENGTH_(_0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14, _15, _16, _17, _18, _19, _20, N, ...) N
#define VA_LENGTH(...) VA_LENGTH_(0, ## __VA_ARGS__, 20, 19, 18, 17, 16, 15, 14, 13, 12, 11, 10, 9, 8, 7, 6, 5, 4, 3, 2, 1, 0)
#define ExecVF(Func, ...) Func(VA_LENGTH(__VA_ARGS__), __VA_ARGS__)
现在您应该像这样使用 ExecVF:
int
main(int argc, char **argv) {
ExecVF(f, "foo", "bar");
ExecVF(f, "foo");
return 0;
}
我需要区分(重载)两个函数 - 一个接受单个 const char*
参数,另一个接受至少两个参数 - const char*
后跟一个或多个参数。
即基本上:
void f(const char *str);
void f(const char *format, ...)
我希望 f("hello")
调用第一个版本,f("hello %d", 10)
调用第二个版本。上面的重载不会工作,因为编译器发现 f("hello")
不明确。
所以我尝试了这个:
void f(const char *str);
template<typename T>
void f(const char *str, T tt, ...);
这使得重载解析工作正常。但我最终遇到了另一个问题。第二个函数应该转发 printf 样式用法的参数。所以我有类似的东西:
template <typename T>
void f ( const char *format, T tt, ... )
{
(T)tt;
va_list varlist;
va_start(varlist, format);
vprintf(format, varlist);
va_end(varlist);
}
现在第二个参数 tt
不再是可变参数列表的一部分,并且使用 format
does not seem to work.
va_start()
有什么方法可以达到我想要的效果吗?
也许使用模板不是最好的主意,因为第二个错误似乎比第一个更难修复。函数的重载很好,如果你在一个单独的文件中尝试,你会发现它更有效。
#include <iostream>
bool f(const char* str) {
return true;
}
bool f(const char* str,int n) {
return false;
}
int main(int argc,char* argv[]) {
std::cout << f("Hi") << std::endl; //Gives 1
std::cout << f("Hi",10) << std::endl; //Gives 0
return 0;
}
您也可以尝试创建一个命名空间并将函数放入其中,从而确保没有其他文件共享可能导致该错误的函数名称。
如果您使用可变参数模板,您可以完成您想要的:
#include <iostream>
void f(const char* str)
{
std::cout << "single arg: " << str << "\n";
}
template <typename ...T>
void f(const char *format, T... args)
{
printf(format, args...);
}
int main()
{
f("hello");
f("formatted %d", 10);
}
- 问:为什么编译出错?
- 答:因为“void f(const char *format, ...)”的范围覆盖了“void f(const char *str);”。
所以可爱的解决方案是你最好使用 arg num 来区分 2 个函数。
像这样:
void func_with_one_arg(const char *str) {
printf("func_with_one_arg");
}
void f(int arg_num, const char *format, ...) {
if (arg_num==1) return func_with_one_arg(format);
printf("func_with_multi_arg\n");
va_list ap;
va_start(ap, format);
va_end(ap);
}
如果你想自动计算变量arg_num,使用这个宏来帮助计算。
#define VA_LENGTH_(_0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14, _15, _16, _17, _18, _19, _20, N, ...) N
#define VA_LENGTH(...) VA_LENGTH_(0, ## __VA_ARGS__, 20, 19, 18, 17, 16, 15, 14, 13, 12, 11, 10, 9, 8, 7, 6, 5, 4, 3, 2, 1, 0)
#define ExecVF(Func, ...) Func(VA_LENGTH(__VA_ARGS__), __VA_ARGS__)
现在您应该像这样使用 ExecVF:
int
main(int argc, char **argv) {
ExecVF(f, "foo", "bar");
ExecVF(f, "foo");
return 0;
}