C++ 可变参数模板

可变参数列表

C语言中的可变参数列表

#include int main(){ printf("%d %d %d", 1, 2, 3);
    return 0;
}

​ ***可变参数列表:***接受任意个数的参数,如 printf

int printf(const char* format, ...);

​ 函数定义形参部分带 … 。在 C 语言中,三个点并非必要。

C++中的可变参数列表

形参包

void f(int ...num);

形参包: … 与形参名结合便是形参包。形参包可接受任意多个参数。

​ ***范式:***上述的 num 便是范式。… 在形参名之前。

void f(int ...num) { int arr[] = {num...};
    //假设传了五个参数,则展开后为
    //int arr[] = {num1, num2, num3, num4, num5};
}

​ ***形参包展开:***参数列表中创建的形参包在后续使用必须展开。… 在范式后即为形参包展开。只展开 … 之前的语句,语句必须带范式形参。展开后的语句以逗号隔开。

可变参数模板

类型形参包

template

​ ***类型形参包:***与形参包类似,… 后带类型模板便是类型形参包。

templatestd::common_type_t Add(T ...num) { /*...*/
}
/*
std::common_type_t:自动适配尖括号中类型形参包的公共类型。
假设调用为:Add(1.2,3,5.5f,'\0',2ULL):
std::common_type_t展开后为:
std::common_type_tAdd(T ...num)展开后为:
Add(double num1, int num2, float num3, char num4, unsign long long num5)
*/

​ ***类型形参包展开:***与形参包展开类似,… 在类型名后便是类型形参包展开。

***总结:***三个点在变量名或模板类型名前是创建形参包,三个点在变量名或模板类型名后是展开形参包。

形参包后续类型形参

#include template>RT Add(const Args& ...nums) {RT temp[] = { nums... };
	return std::accumulate(std::begin(temp), std::end(temp), RT{ 0 });
}
//std::accumulate 通过迭代器求和
//std::accumulate(起始迭代器, 终止迭代器, 求和结果变量)

​ 后续类型形参无法手动指定,只能通过缺省类型进行指定或推导。

推导指引(C++17)

​ 推导指引只作用于类型模板。

templatestruct test {private:
	T a;
public:
	test(T n) :a{ n } {}
	void type() {std::cout << typeid(a).name() << std::endl;
	}
};
//推导指引,将int类型推导为double类型后调用模板
test(int)->test;

利用可变参数模板进行推导指引

//模板A,定义结构体array
templatestruct array {Ty arr[size];
};
//模板B,用于推导指引
templatearray(T, Args...) -> array;
//sizeof...(形参包):形参包参数个数

​ 如果没有模板B,多参数创建 array 必须指定模板类型。

array a{ 1,2,3,4,5 };
array a{ 1,2,3,4,5 };//没有模板B则编译不通过,编译器无法自行推导