c和c++中,实现函数接受可变长参的方法有好几种,本文介绍的是c++中的可变参数模板。
关键要素:
模板参数包、函数参数包、递归
1,可变参数模板函数和普通函数的重载
示例:
#include<iostream>
using namespace std;
void myCout(){}
template<typename A, typename ... T>
void myCout(A a, T... x)
{
cout << a << ',';
myCout(x...);
}
int main()
{
myCout(12, '?', 0.1, "abcd");
return 0;
}
输出:
12,?,0.1,abcd,
改进:
让最后一个参数后面不输出逗号
改进代码:
#include<iostream>
using namespace std;
template<typename A>
void myCout(A a)
{
cout << a;
}
template<typename A, typename ... T>
void myCout(A a, T... x)
{
cout << a << ',';
myCout(x...);
}
int main()
{
myCout(12, '?', 0.1, "abcd");
return 0;
}
2,可变参数模板函数之间的重载
#include<iostream>
using namespace std;
template<typename ... T>
void myCout(T... x)
{
cout<<"end";
}
template<typename A, typename ... T>
void myCout(A a, T... x)
{
cout << a << ',';
myCout(x...);
}
int main()
{
myCout(12, '?', 0.1, "abcd");
return 0;
}
输出:12,?,0.1,abcd,end
这种情况下,更严格的参数如果能匹配上就优先匹配。
3,模板参数包的传递
#include<iostream>
using namespace std;
template<typename ... T>
void myCout(T... x)
{
cout<<"end\n";
}
template<typename A, typename ... T>
void myCout(A a, T... x)
{
cout << a << ',';
myCout(x...);
}
template<typename ... T>
void myCout2(T... x)
{
myCout(x...);
}
int main()
{
myCout2(12, '?', 0.1, "abcd");
myCout2();
return 0;
}
输出:
12,?,0.1,abcd,end
end
4,综合应用
#include<iostream>
using namespace std;
template<typename ... T>
void f1(T... x);
template<typename ... T>
void f3(T... x){};
template<typename A, typename ... T>
void f3(A a, T... x)
{
cout<<"f3 ";
cout << a << ',';
f1(x...);
}
template<typename ... T>
void f2(T... x)
{
cout<<"f2 ";
f3(x...);
}
template<typename ... T>
void f1(T... x)
{
cout<<"f1 ";
f2(x...);
}
int main()
{
f1(12, '?', 0.1, "abcd");
return 0;
}
输出:
f1 f2 f3 12,f1 f2 f3 ?,f1 f2 f3 0.1,f1 f2 f3 abcd,f1 f2