c++11之函数参数包展开
1.关于
本文略带总结性,参考:泛化之美--C++11可变模版参数的妙用
参数包展开方式有两种: 递归展开 和 逗号表达式展开。
本文代码并非全部来自参考文章,自己做了注释和修改。请以原文为准
2. 查询参数包的参数个数
用下面的函数可以实现查询参数包的个数
template<typename... Args>
unsigned int length(Args... args)
{
return sizeof...(args);
}
调用
length(1, "hello", "c++", "11"); // 返回4,4个参数
3.递归展开
3.1 注意, 递归展开需要考虑爆栈的情况。 说到这里,ubuntu(linux) 默认栈大小8M(使用命令 ulimit -a查看), Win下,Visual Studio 默认栈大小1M(连接器->系统)。
3.2 需要两个函数:递归终止函数 和 递归函数
3.3 一个例子,输出各个元素。函数参数为参数包的形式,使用递归展开.
// 1.递归终止函数
void print()
{
;
}
// 2.递归执行函数
template<typename T, typename... Args>
void print(T head, Args... args)
{
std::cout << head << (sizeof...(args) > 0 ? ", " : "");
print(args...);
}
3.4 调用及执行结果
print(1, "hello", "C++", 11);

3.5 其执行过程
print(1, "hello", "C++", 11);
print("hello", "C++", 11);
print("C++", 11);
print(11);
print();
例子中,递归终止到第0参数截止。根据执行过程可知,终止函数可以改为1个参数,最后一个函数仅仅作为终止递归使用,终止递归改为1个参数的函数还可以提高性能(减少函数的调用)。
3.6 递归终止改为一个参数
// 1.递归终止函数
template<typename T>
void print(const T& t)
{
std::cout << t << endl;
}
// 2.递归执行函数
template<typename T, typename... Args>
void print(T head, Args... args)
{
std::cout << head << (sizeof...(args) > 0 ? ", " : "");
print(args...);
}
同样的,执行上面的调用代码print(1, "hello", "C++", 11);,其执行过程如下
print(1, "hello", "C++", 11);
print("hello", "C++", 11);
print("C++", 11);
print(11);
3.7 再看一个递归求和的例子, 同样是使用参数包的形式
// 1.递归终止函数
template <typename T>
T sum(const T& t)
{
return t;
}
// 2.递归函数
template <typename T, typename ... Args>
T sum(const T& head, const Args... args)
{
return head + sum<T>(args...);
}
4.逗号表达式展开
逗号表达式?了解一下
4.1 回到上面print的例子,改为逗号表达式
// 1.用作输出每个参数
template<typename T>
void show_arg(const T& t)
{
std::cout << t <<std::endl;
}
// 2.用作展开参数包
template <typename ... Args>
void expand(const Args&... args)
{
int arr[] = {(show_arg(args), 0)...}; /// (1)
}
代码中,(1)的逗号表达式为: (show_arg(args), 0)。先执行表达式1(show_arg(args)),再执行表达式2(0)。 使用调用代码expand(1, 2, 3, 4),(1)处的逗号表达式展开结果如下
int arr[] = { (show_arg(1), 0) , (show_arg(2), 0), (show_arg(3), 0), (show_arg(4), 0) };
而,第一个逗号表达式结果为0, 第二逗号表达式的结果为0, .... 最后一个逗号表达式展开后,逗号表达式的结果为0,所以(1)的代码最后是这样的:
int arr[] = {0, 0, 0, 0}; /// (2)
(2)的代码声明了一个int数组,数组有4个元素。 展开参数包的方式相当巧妙,比起递归,更加方便了。
4.2 使用lambda + 逗号表达式展开参数包
还是同样的例子,输出参数包元素。
template<typename Func, typename...Args>
void show_args(const Func &f, Args&... args)
{
// 初始化列表,列表的类型为int, std::initializer_list是c++11引入的模板
std::initializer_list<int>{(f(std::forward< Args>(args)), 0)...};
}
// 调用
show_args([](const int & item) {std:: cout << item << ", "; }, 1, 2, 3, 4); /// 输出: 1, 2, 3, 4,
c++11之函数参数包展开的更多相关文章
- C++11变长参数模板
[C++11变长参数模板] C++03只有固定模板参数.C++11 加入新的表示法,允许任意个数.任意类别的模板参数,不必在定义时将参数的个数固定. 实参的个数也可以是 0,所以 tuple<& ...
- 泛化之美--C++11可变模版参数的妙用
1概述 C++11的新特性--可变模版参数(variadic templates)是C++11新增的最强大的特性之一,它对参数进行了高度泛化,它能表示0到任意个数.任意类型的参数.相比C++98/03 ...
- -C++11可变模版参数(转载)
泛化之美--C++11可变模版参数的妙用 1概述 C++11的新特性--可变模版参数(variadic templates)是C++11新增的最强大的特性之一,它对参数进行了高度泛化,它能表示0到任意 ...
- c++之可变参数格式化字符串(c++11可变模板参数)
本文将使用 泛型 实现可变参数. 涉及到的关见函数: std::snprintf 1.一个例子 函数声明及定义 1 // 泛型 2 template <typename... Args> ...
- [C++11]C++可变参数模板
可变参数模板 原文链接: http://blog.csdn.net/xiaohu2022/article/details/69076281 普通模板只可以采取固定数量的模板参数.然而,有时候我们希望模 ...
- ES6之函数参数
ES6中对于函数参数主要增加了以下内容: 1.参数的扩展/数组的展开: 2.默认参数. 什么是参数的扩展? 看下面代码: <!DOCTYPE html> <html lang=&qu ...
- 2、Golang基础--包的使用、if-else语句、循环、switch语句、数组、切片、可变函数参数、map类型
1 包的使用 // 为了便于组织代码,同一种类型的代码,写在同一个包下,便于管理 // 定义包 -新建一个文件夹 -内部有很多go文件 -在每个go文件的第一行,都要声明包名,并且包名必须一致 -在一 ...
- 用C++11实现C++17的apply(动态数组用作函数参数)
标题有点错误,apply是用tuple做参数,调用一个函数.这个标题是为了能更好的适配搜索关键字. 动态数组用作函数参数更适合嵌入了脚本环境的C++程序,比如lua或javascript(js). 若 ...
- Oracle中如何导出存储过程、函数、包和触发器的定义语句?如何导出表的结构?如何导出索引的创建语句?
Oracle中如何导出存储过程.函数.包和触发器的定义语句?如何导出表的结构?如何导出索引的创建语句? QQ群里有人问:如何导出一个用户下的存储过程? 麦苗答:方法有多种,可以使用DBMS_MET ...
随机推荐
- 多组学分析及可视化R包
最近打算开始写一个多组学(包括宏基因组/16S/转录组/蛋白组/代谢组)关联分析的R包,避免重复造轮子,在开始之前随便在网上调研了下目前已有的R包工具,部分罗列如下: 1. mixOmics 应该是在 ...
- kubernetes 用到的工具及组件
kubernetes 用到的工具及组件,将所有的组件下载后放到/usr/local/bin目录下(记得chmod a+x /usr/local/bin/*).所有的组件,原则上都用最新的,如果遇到不支 ...
- MPI 学习笔记
目录 MPI学习笔记 MPI准备 概述 前置知识补充 环境部署 1.修改IP及主机名 2.关闭防火墙 3.实现免密码SSH登录 4.配置MPI运行环境 5.测试 程序的执行 编译语句 运行语句 MPI ...
- 网口程序udp
# -*- coding: utf-8 -*- """ Created on Thu Nov 12 15:02:53 2020 @author: Administrato ...
- 【编程思想】【设计模式】【创建模式creational】lazy_evaluation
Python版 https://github.com/faif/python-patterns/blob/master/creational/lazy_evaluation.py #!/usr/bin ...
- OceanBase 2.x体验:推荐用DBeaver工具连接数据库
Original MQ4096 [OceanBase技术闲谈](javascript:void(0) 2020-01-15 OceanBase 2.x体验:推荐用DBeaver工具连接数据库 Ocea ...
- shell脚本 用户登录服务器发送钉钉提醒
一.企业微信配置 1.获取AgentId(AppID).Secret .CropID.部门ID 创建一个企业微信应用获取到AgentId(AppID).Secret 2.获取CropID,点击 &qu ...
- Three.js 实现3D全景侦探小游戏🕵️
背景 你是嘿嘿嘿侦探社实习侦探️,接到上级指派任务,到甄开心小镇调查市民甄不戳宝石失窃案,根据线人流浪汉老石提供的线索,小偷就躲在小镇,快把他找出来,帮甄不戳寻回失窃的宝石吧! 本文使用 Three ...
- 从 CPython 源码角度看 Python 垃圾回收机制
环状双向链表 refchain 在 Python 程序中创建的任何对象都会被放到 refchain 链表中,当创建一个 Python 对象时,内部实际上创建了一些基本的数据: 上一个对象 下一个对象 ...
- [BUUCTF]PWN——bbys_tu_2016
bbys_tu_2016 附件 步骤: 例行检查,32位程序,开启了nx保护 本地试运行一下程序,看看大概的情况,测试时候发现输入长度过长程序会崩溃,猜测输入点存在问题 32位ida载入,检索程序里的 ...