c++11边长模板参数

#include <iostream>
#include <cstdlib>

void print() {  // 这里结束模板递归地终止链
    std::cout << "I am empty function and I am called at last.\n";
}

template<typename T, typename... Types>
void print(T var1, Types... var2) {
    std::cout << var1 << std::endl;
    std::cout<<"sizeof...(var2)="<<sizeof...(var2)<<endl;//计算剩下的参数个数,如果参数为0,           则会调用print(void)
    print(var2...);
}

int main() {
    print(1,2,3.14, "Pass me any number of arguments",
        "I will print\n");
    system("pause");
    return 0;
}

计算变长参数个数,如上面注释:sizeof...(var2) 可以计算var2变长参数剩余个数

边长参数模板相当于一个模板的递归展开模型,但是它不是递归的,效率和循环的类似。使用的时候,要定义一个“递归”的出口,然后定义一系列的操作,操作的是以“递归”的方式进行的。

函数的调用过程类似于一个递归展开,先执行std::cout << var1 << std::endl;,然后递归地向下展开执行print(var2...);,此时的参数包也进行解包操作。

对于引用类型的,也是相似的操作,注意引用的位置和const的位置即可。
 

#include <iostream>
#include <cstdlib>
#include <string>

void log() {
    // 终止模板递归链
}

template<typename T, typename ...Args>
void log(const T& t, const Args&... params) {
    std::cout << t << std::endl;
    log(params...);  // 向下递归
}

int main() {
    log(1.22, 19, "hello world !");
    system("pause");
    return 0;
}

一般情况下参数包必须在最后面,例如:

template <typename T, typename ... Args> void fun(T t,Args ... args);//合法

template <typename ... Args, typename T> void fun(Args ... args,T t);//非法

2,解包 (包展开)

在实际使用时,拿到一个复合而成的包对没有并没有什么用,我们通常需要获得它里面内一个元素的内容。解包是把参数包展开为它所表示的具体内容的动作。

解包时采用“包扩展表达式”,就是包名加上三个点,如“Args...”。

例如:

假设我们有一个模板类Base:

template <typename ... Args>
class D1 : public Base<Args...>{};

template <typename ... Args>
class D2 : public Base<Args>...{};

解包用两种常见的形式:

1,直接解包(上面第一个)

D1<X,Y,Z> 相当于 D1:public Base<X,Y,Z>

2,先参与其他表达式再解包(上面第二个)

D2<X,Y,Z> 相当于 D2: public Base<X>, Base<Y>, Base<Z>


直观上理解就是在...所在的位置将包含了参数包的表达式展开为若干个具体形式。
 

template<typename...T>
void DummyWrapper(const T&... t){}


template<typename M>
auto  unpacker(const M& t)->decltype(t){
	cout << "," << t;
	return t;
}
template<typename T,typename... Args>
void write_line(const T& t, const Args&... data) {
	cout << t;
	DummyWrapper(unpacker(data)...);
	cout << endl;
}

write_line(1, 2, 3.14, "Pass me any number of arguments", "I will print");

出现一个问题:如果把typename M改为typename T,出现编译错误,原因是,匹配不到正确的函数,这个是因为生成了一份upacker特化,所以找不到函数,如果换成typename M,会生成对应的类型,这可能与typename作用域范围有关

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值