可变参数模板

原文链接: http://blog.csdn.net/xiaohu2022/article/details/69076281
普通模板只可以采取固定数量的模板参数。然而,有时候我们希望模板可以接收任意数量的模板参数,这个时候可以采用可变参数模板。对于可变参数模板,其将包含至少一个模板参数包,模板参数包是可以接收0个或者多个参数的模板参数。相应地,存在函数参数包,意味着这个函数参数可以接收任意数量的参数。

使用规则

一个可变参数类模板定义如下:

template<typename ... Types>
class Tuple
{};

可以用任意数量的类型来实例化Tuple:

Tuple<> t0;
Tuple<int> t1;
Tuple<int, string> t2;
// Tuple<0> error; 0 is not a type

如果想避免出现用0个模板参数来实例化可变参数模板,可以这样定义模板:

template<typename T, typename ... Types>
class Tuple
{};

此时在实例化时,必须传入至少一个模板参数,否则无法编译。
同样地,可以定义接收任意参数的可变参数函数模板:

template<typename ... Types>
void f(Types ... args); // 一些合法的调用
f();
f();
f(3.4, "hello");

对于类模板来说,可变模板参数包必须是模板参数列表中的最后一个参数。但是对于函数模板来说,则没有这个限制,考虑下面的情况:

template<typename ... Ts, typename U>
class Invalid
{}; // 这是非法的定义,因为永远无法推断出U的类型 template<typename ... Ts, typename U>
void valid(U u, Ts ... args); // 这是合法的,因为可以推断出U的类型
// void invalid(Ts ... args, U u); // 非法的,永远无法推断出U valid(1.0, , , ); // 此时,U的类型是double,Ts是{int, int, int}

可变参数函数模板实例

无法直接遍历传给可变参数模板的不同参数,但是可以借助递归的方式来使用可变参数模板。可变参数模板允许创建类型安全的可变长度参数列表。下面定义一个可变参数函数模板processValues(),它允许以类型安全的方式接受不同类型的可变数目的参数。函数processValues()会处理可变参数列表中的每个值,对每个参数执行对应版本的handleValue()。

// 处理每个类型的实际函数
void handleValue(int value) { cout << "Integer: " << value << endl; }
void handleValue(double value) { cout << "Double: " << value << endl; }
void handleValue(string value) { cout << "String: " << value << endl; } // 用于终止迭代的基函数
template<typename T>
void processValues(T arg)
{
handleValue(arg);
} // 可变参数函数模板
template<typename T, typename ... Ts>
void processValues(T arg, Ts ... args)
{
handleValue(arg);
processValues(args ...); // 解包,然后递归
}

可以看到这个例子用了三次... 运算符,但是有两层不同的含义。用在参数模板列表以及函数参数列表,其表示的是参数包。前面说到,参数包可以接受任意数量的参数。用在函数实际调用中的...运算符,它表示参数包扩展,此时会对args解包,展开各个参数,并用逗号分隔。模板总是至少需要一个参数,通过args...解包可以递归调用processValues(),这样每次调用都会至少用到一个模板参数。对于递归来说,需要终止条件,当解包后的参数只有一个时,调用接收一个参数模板的processValues()函数,从而终止整个递归。

假如对processValues()进行如下调用:

processsValues(, 2.5, "test");

其产生的递归调用如下:

processsValues(, 2.5, "test");
handleValue();
processsValues(2.5, "test");
handleValue(2.5);
processsValues("test");
handleValue("test");

由于processValues()函数会根据实际类型推导自动调用正确版本的handleValue()函数,所以这种可变参数列表是完全类型安全的。如果调用processValues()函数带有的一个参数,无对应的handleValue()函数版本,那么编译器会产生一个错误。

前面的实现有一个致命的缺陷,那就是递归调用时参数是复制传值的,对于有些类型参数,其代价可能会很高。一个高效且合理的方式是按引用传值,但是对于字面量调用processValues()这样会存在问题,因为字面量仅允许传给const引用参数。比较幸运的是,我们可以考虑右值引用。使用std::forward()函数可以实现这样的处理,当把右值引用传递给processValues()函数时,它就传递为右值引用,但是如果把左值引用传递给processValues()函数时,它就传递为左值引用。下面是具体实现:

// 用于终止迭代的基函数
template<typename T>
void processValues(T &&arg)
{
handleValue(std::forward<T>(arg));
} // 可变参数函数模板
template<typename T, typename ... Ts>
void processValues(T&& arg, Ts&& ... args)
{
handleValue(std::forward<T>(arg));
processValues(std::forward<Ts>(args) ...); // 先使用forward函数处理后,再解包,然后递归
}

实现简化的printf函数

这里我们通过可变参数模板实现一个简化版本的printf函数:

// 基函数
void tprintf(const char* format)
{
cout << format;
} template<typename T, typename ... Ts>
void tprintf(const char* format, T&& value, Ts&& ... args)
{
for (; *format != '\0'; ++format)
{
if (*format == '%')
{
cout << value;
tprintf(format + , std::forward<Ts>(args) ...); // 递归
return;
}
cout << *format;
}
}
int main()
{ tprintf("% world% %\n", "Hello", '!', );
// output: Hello, world! 2017
cin.ignore();
return ;
}

其方法基本与processValues()是一致的,但是由于tprintf的第一个参数固定是const char*类型。

References

[1] Marc Gregoire. Professional C++, Third Edition, 2016.
[2] cppreference parameter pack

原文作者:小白将
链接:https://www.jianshu.com/p/4bf4d1860588

[C++11]C++可变参数模板的更多相关文章

  1. C++11变长参数模板

    [C++11变长参数模板] C++03只有固定模板参数.C++11 加入新的表示法,允许任意个数.任意类别的模板参数,不必在定义时将参数的个数固定. 实参的个数也可以是 0,所以 tuple<& ...

  2. c++ 可变参数模板

    可变参数模板,自己尝试了个例子,如下: // variadicTemplates.cpp : 定义控制台应用程序的入口点. // #include "stdafx.h" #incl ...

  3. C++11新特性之五——可变参数模板

    有些时候,我们定义一个函数,可能这个函数需要支持可变长参数,也就是说调用者可以传入任意个数的参数.比如C函数printf(). 我们可以这么调用. printf(); 那么这个函数是怎么实现的呢?其实 ...

  4. c++11——可变参数模板

    在c++11之前,类模板和函数模板只能含有固定数量的模板参数,c++11增加了可变模板参数特性:允许模板定义中包含0到任意个模板参数.声明可变参数模板时,需要在typename或class后面加上省略 ...

  5. C++11 可变参数模板

    在C++11之前, 有两个典型的受制于模板功能不强而导致代码重复难看的问题, 那就 function object 和 tuple. 拿 function objects 来说, 需要一个返回类型参数 ...

  6. C学习笔记(11)--- 可变参数,浅谈内存管理 【C基础概念系列完结】

    1.可变参数(variable arguments): 可变参数允许您定义一个函数,能根据具体的需求接受可变数量的参数. int func(int, ... )             (函数 fun ...

  7. C++ 标准库,可变参数模板。可变参数数量,可变参数类型【转】

    #include <iostream> // 可变模板参数 // 此例:可以构造可变数量,可变类型的函数输入. // 摘自:https://www.cnblogs.com/qicosmos ...

  8. C++11实现可变参数泛型抽象工厂

  9. C++学习之可变参数的函数与模板

    所谓可变参数指的是函数的参数个数可变,参数类型不定的函数.为了编写能处理不同数量实参的函数,C++11提供了两种主要的方法:如果所有的实参类型相同,可以传递一个名为initializer_list的标 ...

随机推荐

  1. Linux进程的五个段

    目录 数据段 代码段 BSS段 堆(heap) 栈 数据段 用来存放可执行文件中已初始化的全局变量,换句话说就是存放程序静态分配的变量和全局变量: 代码段 代码段是用来存放可执行文件的操作指令,也就是 ...

  2. 协议——SCCB与IIC的区别

    SCCB(Serial Camera Control Bus,串行摄像头控制总线)是由OV(OmniVision的简称)公司定义和发展的三线式串行总线,该总线控制着摄像头大部分的功能,包括图像数据格式 ...

  3. centos7+ 在线yum安装docker-ce

    yum install -y yum-utils   //扩展yum功能 yum-config-manager --add-repo http://mirrors.aliyun.com/docker- ...

  4. Golang --多个变量同时赋值

    编程最简单的算法之一,莫过于变量交换.交换变量的常见算法需要一个中间变量进行变量的临时保存.用传统方法编写变量交换代码如下: var a int = 100 var b int = 200 var t ...

  5. java8之lambda表达式(默认方法)

    [推荐]2019 Java 开发者跳槽指南.pdf(吐血整理)>>> 许多开发语言都将函数表达式集成到了其集合库中.这样比循环方式所需的代码更少,并且更加容易理解.以下面的循环为例: ...

  6. Hadoop1-认识Hadoop大数据处理架构

    一.简介概述 1.什么是Hadoop Hadoop是Apache软件基金会旗下的一个开源分布式计算平台,为用户提供了系统底层细节透明的分布式基础架构 Hadoop是基于java语言开发,具有很好的跨平 ...

  7. git重置账号密码

    1.打开控制面板(快捷打开win+R,输入control) 2.点击打开用户账户 3.点击凭据管理器 4.点击windows凭据删除你的git凭据即可

  8. .Net Core 注入学习——注册服务

    解析 .Net Core 注入——注册服务发表于:2017-10-23 10:47 作者:行走即歌 来源:51Testing软件测试网采编字体:大 中 小 | 上一篇 | 下一篇 |我要投稿 | 推荐 ...

  9. php后台实现页面跳转的方法-转载

    地址:http://blog.csdn.net/abandonship/article/details/6459104 其中方法三的js代码在tp框架使用存在故障,一个是需要把代码写在一起(可能也不需 ...

  10. HTML学习摘要2

    DAY 2 HTML 标签可以拥有属性.属性提供了有关 HTML 元素的更多的信息. 属性总是以名称/值对的形式出现,比如:name="value". 属性总是在 HTML 元素的 ...