第四章 可变参数模板

简单示例

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");
}

可变参数函数模板参数与普通参数适用相同的规则。通过值传递,参数复制并衰变 (例如,数 组变成指针),若通过引用传递,参数指向原始参数,而不衰变

应用

  1. 比如,我想将所有参数包(类型可能不一致)都各自相加一次,然后输出:
#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"));
}
  1. 给参数包的每个元素都 加上一个值:
#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: 不得不说,这里的语法很抽象。。。

  1. 判断参数包中的所有元素是否全都是同一个参数类型(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;
}
  1. 参数包当作索引,访问数组元素:

#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);
}
  1. 类模板做索引
#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… 可为参数包提供的参数数量。
• 可变参数模板的一个应用是转发任意数量的类型参数。
• 通过使用折叠表达式,可以对参数包中的所有参数使用相应的操作符。

  • 7
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值