博客主页: 主页
系列专栏: C++
❤️感谢大家点赞收藏⭐评论✍️
期待与大家一起进步!
可变参数模板的使用:
template<class T>
void ShowList(T value) {
cout << value<<" ";
}
template<class T,class ...Args>
void ShowList(T value, Args...args) {
cout << value << " ";
ShowList(args...);
}
// Args是一个模板参数包,args是一个函数形参参数包
// 声明一个参数包Args...args,这个参数包中可以包含0到任意个模板参数。
int main()
{
ShowList(1);
cout << endl;
ShowList(1, 'A');
cout << endl;
ShowList(1, 'A', std::string("sort"));
cout << endl;
return 0;
}
上面的参数args前面有省略号,所以它就是一个可变模版参数,我们把带省略号的参数称为“参数包”,它里面包含了0到N(N>=0)个模版参数。我们无法直接获取参数包args中的每个参数的,只能通过展开参数包的方式来获取参数包中的每个参数,这是使用可变模版参数的一个主要特点,也是最大的难点,即如何展开可变模版参数。由于语法不支持使用args[i]这样方式获取可变参数
当递归到参数包里面只有一个参数的时候会去调用第一个ShowList(T value)函数
template <class T>
void PrintArg(T t)
{
//printarg不是一个递归终止函数,只是一个处理参数包中每一个参数的函数。
cout << t << " ";
}
//展开函数
template <class ...Args>
void ShowList(Args... args)
{
int arr[] = { (PrintArg(args), 0)... };
//先执行printarg(args),再得到逗号表达式的结果0
//最终会创建一个元素值都为0的数组
//由于是逗号表达式,在创建数组的过程中会先执行逗号表达式前面的部分printarg(args)
//打印出参数,也就是说在构造int数组的过程中就将参数包展开了,这个数组的目的纯粹是为了在
//数组构造的过程展开参数包
cout << endl;
}
int main()
{
ShowList(1);
ShowList(1, 'A');
ShowList(1, 'A', std::string("sort"));
return 0;
}
我们看到的emplace系列的接口,支持模板的可变参数,并且万能引用
这是因为使用这种方式某种程度上可以提高效率
emplace直接自己去构建pair中的类型,省去了string中还要进行交换的过程。