★10.关于可变参数模板

2017-06-30  本文已影响0人  iDragonfly

可变参数函数模板

// Args称为模板参数包,rest称为函数参数包
template <typename T, typename ... Args>
void fun(T t, const Args & ... rest) {
}

可变参数类模板

// Args称为模板参数包,rest称为函数参数包
template <typename T, typename ... Args>
class A {
};

int main() {
    A<int, int, double> a;
    return system("pause");
}

可变参数函数模板的使用

// Args称为模板参数包,rest称为函数参数包
template <typename T, typename ... Args>
void fun(T t, const Args & ... rest) {
    // 获取模板参数包中的参数个数
    std::cout << sizeof...(Args) << std::endl;
    // 获取函数参数包中的参数个数
    std::cout << sizeof...(rest) << std::endl;

    // 可以利用递归处理每一个函数参数包中的参数
    // 相比上面的函数参数少了一个参数,可以知道函数参数包里的参数会一个一个减少
    fun(rest...);                          // rest...代表函数包拓展,代表着拓展为函数实参列表
//    fun(change(rest)...);                  // 更复杂的函数包拓展,每个参数都先用change函数处理一下并返回
};

// 巧妙点:上述的递归中,当最后一个参数时会调用更特例化版本,避免了模板参数包为0的报错
template <typename LastOne>
void fun(const LastOne & lastOne) {
    std::cout << "The End!" << std::endl;
}

template <typename T>
int change(T t) {
    return 0;               // 想作为包拓展的模式,必须要有返回值
}

int main() {
    int i1 = 0;
    double i2 = 0;
    char i3 = 0;
    float i4 = 0;
    // 备注:模板参数包的参数个数不可以为0,以下调用模板参数包的个数为4
    fun(i1, i2, i3, i4);    // fun<int, double, char, float>
    return system("pause");
}

转发参数包

// 转发参数包
template <typename ... Args>
void fun1(Args && ... rest) {
    // 转发参数包给fun2(),此处非递归调用只是将所有的参数包简单地转发了一下
    fun2(std::forward<Args>(rest)...);             // 同时应用了模板参数包拓展和函数参数包拓展
}

template <typename T, typename ... Args>
void fun2(T t, Args ... rest) {
    fun2(rest...);                              // 递归调用,函数参数包会一个一个减少
};

// 当只有一个参数时,会调用这个更特例化的版本
template <typename T>
void fun2(T t) {
    std::cout << "The End!" << std::endl;
}
上一篇下一篇

猜你喜欢

热点阅读