简单示例
C++11 新特性,可以使用可变数量的模板参数
#include <iostream>
void print() {} // 递归基, 也就是递归的尽头
// 递归函数, T 是参数类型, ... 是参数列表
template<typename T, typename... Args>
void print(T first_arg, Args... args) {
std::cout << first_arg << " \n"; // 打印第一个参数
print(args...); // 递归调用,传入剩下的参数列表
}
int main() {
print(1, 2.2, "hello");
}
或者,重载一个单模板参数的模板函数:
#include <iostream>
template<typename T>
void print(T arg) {
std::cout<<arg<<std::endl;
}
// 递归函数, T 是参数类型, ... 是参数列表
template<typename T, typename... Args>
void print(T first_arg, Args... args) {
print(first_arg); // 调用单参数版本的print
print(args...); // 将模板参数列表传递给print
}
int main() {
print(1, 2.2, "hello");
}
C++11 还为可变参模板引入了一种新的操作符:sizeof…,其值为参数包包含的元素数量:
#include <iostream>
template<typename T>
void print(T arg) {
std::cout<<arg<<std::endl;
}
// 递归函数, T 是参数类型, ... 是参数列表
template<typename T, typename... Args>
void print(T first_arg, Args... args) {
std::cout<<"现在参数列表还有"<<sizeof...(args)<<"个参数\n";
print(first_arg); // 调用单参数版本的print
print(args...); // 将模板参数列表传递给print
}
int main() {
print(1, 2.2, "hello");
}
折叠表达式
C++17新特性, 可以对参数包(列表)中的所有参数使用二元运算符计算结果,前提是参数包不为空(除非是空参数包进行 && 或者 || 操作):
#include <iostream>
template<typename... Args>
auto Sum(Args... args) {
return (args + ...); // 等价于 (... + args)
}
int main() {
std::cout<<Sum(1, 2.2, 3.1)<<std::endl; // 计算综合
std::cout<<Sum()<<std::endl; // 错误,Unary fold expression has empty expansion for operator '+' with no fallback value
}
#include <iostream>
template<typename... Args>
auto Sum(Args... args) {
return (args + ... + 3); // 带init的折叠表达式,从3开始加
}
int main() {
std::cout<<Sum(1, 2.2, 3.3)<<std::endl; // 结果为 false
}
#include <iostream>
template<typename... Args>
auto Sum(Args... args) {
return (args && ...);
}
int main() {
std::cout<<Sum()<<std::endl; // 结果为 true
}
#include <iostream>
template<typename... Args>
auto Sum(Args... args) {
return (... || args);
}
int main() {
std::cout<<Sum()<<std::endl; // 结果为 false
}
还可以打印参数包中的所有参数:
#include <iostream>
// 封装一下,保证每个函数的参数后面都加一个空格
template<typename T>
class AddSpace
{
private:
T const& ref; // refer to argument passed in constructor public:
public:
explicit AddSpace(T const& r): ref(r) {}
friend std::ostream& operator<< (std::ostream& os, AddSpace<T> s) {
return os << s.ref << ' '; // output passed argument and a space
}
};
template<typename... Args>
void Print(const Args&... args) {
(std::cout << ... << AddSpace(args) ) << std::endl;
}
int main() {
Print(1, 2.2, "name");
}
可变参数函数模板参数与普通参数适用相同的规则。通过值传递,参数复制并衰变 (例如,数 组变成指针),若通过引用传递,参数指向原始参数,而不衰变
应用
- 比如,我想将所有参数包(类型可能不一致)都各自相加一次,然后输出:
#include <iostream>
// 封装一下,保证每个函数的参数后面都加一个空格
template<typename T>
class AddSpace
{
private:
T const& ref; // refer to argument passed in constructor public:
public:
explicit AddSpace(T const& r): ref(r) {}
friend std::ostream& operator<< (std::ostream& os, AddSpace<T> s) {
return os << s.ref << '\n'; // output passed argument and a space
}
};
template<typename... Args>
void Print(const Args&... args) {
(std::cout << ... << AddSpace(args) ) << std::endl;
}
// 每个参数都自己和自己相加,然后输出
template<typename... Args>
void DoublePrint(const Args&... args) {
Print(args + args...); // 相当于 Print(1+1, 2.2 + 2.2, "name" + "name");
}
int main() {
// 2 4.4 namename
DoublePrint(1, 2.2, std::string("name"));
}
- 给参数包的每个元素都 加上一个值:
#include <iostream>
// 封装一下,保证每个函数的参数后面都加一个空格
template<typename T>
class AddSpace
{
private:
T const& ref; // refer to argument passed in constructor public:
public:
explicit AddSpace(T const& r): ref(r) {}
friend std::ostream& operator<< (std::ostream& os, AddSpace<T> s) {
return os << s.ref << '\n'; // output passed argument and a space
}
};
template<typename... Args>
void Print(const Args&... args) {
(std::cout << ... << AddSpace(args) ) << std::endl;
}
// 每个参数都自己和自己相加,然后输出
template<typename... Args>
void AddOnePrint(const Args&... args) {
Print(args + 1 ...); // 正确, 1 和 ... 要空一行
// Print(args + 1...); // 语法错误
// Print((args + 1)...); // 正确
}
int main() {
// 2 4.4 namename
AddOnePrint(1, 2.2, 3);
}
ps: 不得不说,这里的语法很抽象。。。
- 判断参数包中的所有元素是否全都是同一个参数类型(C++17 only):
#include <iostream>
template<typename T1, typename... TN>
constexpr bool isHomogeneous (T1, TN...) {
return (std::is_same_v<T1,TN> && ...); // 只要参数包内所有类型相同,返回true
}
int main() {
// 2 4.4 namename
std::cout << isHomogeneous(2,4.4,"namename") << std::endl;
std::cout << isHomogeneous(1,2,3,4,5) << std::endl;
}
- 参数包当作索引,访问数组元素:
#include <iostream>
#include <string>
#include <vector>
// 封装一下,保证每个函数的参数后面都加一个空格
template<typename T>
class AddSpace
{
private:
T const& ref; // refer to argument passed in constructor public:
public:
explicit AddSpace(T const& r): ref(r) {}
friend std::ostream& operator<< (std::ostream& os, AddSpace<T> s) {
return os << s.ref << '\n'; // output passed argument and a space
}
};
template<typename... Args>
void Print(const Args&... args) {
(std::cout << ... << AddSpace(args) ) << std::endl;
}
// 每个参数都自己和自己相加,然后输出
template<typename T, typename... Index>
void printElems(const T& coll, Index... idx) {
Print(coll[idx]...);
}
int main() {
std::vector<std::string> arr{"accd", "efg", "hijk", "lmn"};
printElems(arr, 0, 2, 3);
printElems(arr, 1.9, 2.1, 3.5); // 如果不是整型,会直接取整成[1,2,3]
}
当然,还有另外一种写法,直接使用非类型的参数模板:
#include <iostream>
#include <string>
#include <vector>
// 封装一下,保证每个函数的参数后面都加一个空格
template<typename T>
class AddSpace
{
private:
T const& ref; // refer to argument passed in constructor public:
public:
explicit AddSpace(T const& r): ref(r) {}
friend std::ostream& operator<< (std::ostream& os, AddSpace<T> s) {
return os << s.ref << '\n'; // output passed argument and a space
}
};
template<typename... Args>
void Print(const Args&... args) {
(std::cout << ... << AddSpace(args) ) << std::endl;
}
template<std::size_t... Index, typename T>
void printElems(const T& coll) {
Print(coll[Index]...);
}
int main() {
std::vector<std::string> arr{"accd", "efg", "hijk", "lmn"};
printElems<0,2,3>(arr);
}
- 类模板做索引
#include <iostream>
#include <string>
#include <tuple>
#include <utility>
#include <vector>
#include <algorithm>
// 封装一下,保证每个函数的参数后面都加一个空格
template<typename T>
class AddSpace
{
private:
T const& ref; // refer to argument passed in constructor public:
public:
explicit AddSpace(T const& r): ref(r) {}
friend std::ostream& operator<< (std::ostream& os, AddSpace<T> s) {
return os << s.ref << '\n'; // output passed argument and a space
}
};
template<typename... Args>
void Print(const Args&... args) {
(std::cout << ... << AddSpace(args) ) << std::endl;
}
template<std::size_t...>
struct Indices{
};
template<typename T, std::size_t... Index>
void printByIndex(T t, Indices<Index...> /*unused*/) {
Print(std::get<Index>(t)...);
// Print(t[Index]...); // 同上
}
int main() {
std::array<std::string, 4> arr{"accd", "efg", "hijk", "lmn"};
printByIndex(arr, Indices<0, 2, 3>());
}
总结
• 通过使用参数包,可以为任意数量、类型的模板参数定义模板。 • 要处理参数,需要递归和/或匹配的非变参函数。
• 操作符 sizeof… 可为参数包提供的参数数量。
• 可变参数模板的一个应用是转发任意数量的类型参数。
• 通过使用折叠表达式,可以对参数包中的所有参数使用相应的操作符。