C/C++中的可变参数和可变参数模板
1、说明
不谈官方定义,就从个人理解上说,可变参数 就是函数传参的时候,不确定传入参数的数量和类型,从而动态地在函数内部处理,优点是,函数调用时比较灵活
2、C语言中的可变参数
C语言中一般使用宏定义实现可变参数,先看一个示例:
#include <stdarg.h>
void func(const char *fmt, ...)
{
va_list ap;
va_start(ap, fmt);
auto a = va_arg(ap, int);
auto b = va_arg(ap, double);
auto c = va_arg(ap, char*);
cout << a << ", " << b << ", " << c << endl;
va_end(ap);
}
int main()
{
func("%d %f %s\n", 1, 2.0f, "hello world");
return 0;
}
这是一个很常见的C语言的可变参数的使用,va_start 用于初始化 va_list 变量,va_arg 用于提取可变参数,va_end 用于释放 va_list
这个示例可以用在一般函数上,无法使用在宏定义中,如果一定要在宏定义中使用,需要配合 __VA_ARGS__,示例如下:
//#define CALC(fmt, ...) func(fmt, ...) //错误的使用
#define CALC(fmt, ...) func(fmt, __VA_ARGS__)
int main()
{
CALC("%d %f %s\n", 1, 2.0f, "hello world");
return 0;
}
3、C++中的可变参数模板
C++11 中引入了新的功能,可变参数模版,语法如下:
template <typename T, typename ... Args>
void func(T t,Args ... args);
这里面,Args 称之为模板参数包(template parameter pack),表示模板参数位置上的变长参数,
args 称之为函数参数包(function parameter pack),表示函数参数位置上的变长参数
可以使用 sizeof...() 获取可变参数数目
先看一个示例:
template<typename... Args>
void print(Args... args)
{
int num = sizeof...(args);
}
int main()
{
print(1, 2, "123", 4);
return 0;
}
执行结果是:
4
2.1、使用递归的方式遍历
可变参数一般使用递归的方式进行遍历,利用模板的推导机制,每次从可变参数中取出第一个元素,直到包为空
缺点:递归毕竟是使用栈内存,过多的递归层级容易导致爆栈的发生
示例代码如下:
void printf()
{
cout << "end" << endl;
}
template<typename T, typename... Args>
void print(const T &value, Args... args)
{
cout << value << endl;
print(args...);
}
int main()
{
print(1, 2, "333", 4);
return 0;
}
结果输出如下:
1
2
333
4
end
这里不是很好理解,我们可以自己理解 print() 方法的每一次调用
- main函数中第一次调用,value为1, args有2、"333和4三个值,输出1;
- 第一次递归,即print中调用print,value为2,args有“333”和4两个值,输出2;
- 第二次递归,即print中调用print,value为“333”,args为4,输出“333”;
- 第三次递归,即print中调用print,value为4,args无值,输出4;
- 此时,args因为无值,print(args...) 语句调用的就不再是模板函数,而是第一行的 print(),输出end;
所以,很好理解,为什么要先定义一个同名的函数,就是为了等可变参数经过几次推导之后,没有值的情况出现;
当然,递归遍历也可以这么写:
template<typename T>
void print(T value)
{
cout << "end:" << value << endl;
}
template<typename T, typename... Args>
void print(const T &value, Args... args)
{
cout << value << endl;
print(args...);
}
int main()
{
print(1, 2, "333", 4);
return 0;
}
结果输出:
1
2
333
end:4
这和第一个例子只有些许区别,函数调用如下:
- main函数中第一次调用,value为1, args有2、"333和4三个值,输出1;
- 第一次递归,即print中调用print,value为2,args有“333”和4两个值,输出2;
- 第二次递归,即print中调用print,value为“333”,args为4,输出“333”;
- 此时,args为4,print(args...) 语句调用的就不再是模板函数,而是第一行的 print(4),输出end:4;
2.2、使用非递归的方式遍历
利用 std::initializer_list ,即初始化列表展开可变参数
示例1,使用展开函数处理参数:
template<typename T>
void run(const T &t)
{
cout << t << endl;
}
template<typename... Args>
void print(Args... args)
{
std::initializer_list<int>{(run(args), 0)...};
}
int main()
{
print(1, 2, "333as", 4);
return 0;
}
示例2,使用lambda:
template<typename... Args>
void print(Args... args)
{
std::initializer_list<int>{([&]
{ cout << args << endl; }(), 0)...};
}
int main()
{
print(1, 2, "333as", 4);
return 0;
}
C/C++中的可变参数和可变参数模板的更多相关文章
- python中的函数的参数和可变参数
最近在搞python的过程中需要用到给函数传可变参数..所以去网上找前人的帖子学习了一下 为了尊重原作者,这里附上链接:http://www.cnblogs.com/tqsummer/archive/ ...
- Java中不定项参数(可变参数)的作用和使用方式
引言: 我们在编写方法的过程中,可能会遇见一个方法有不确定参数个数的情况.一般我们会用方法重载来解决问题: //方法重载,解决参数个数不确定问题 public void method(); publi ...
- C++ 11可变参数接口设计在模板编程中应用的一点点总结
概述 本人对模板编程的应用并非很深,若要用一句话总结我个人对模板编程的理解,我想说的是:模板编程是对类定义的弱化. 如何理解“类定义的弱化”? 一个完整的类有如下几部分组成: 类的名称: 类的成员变量 ...
- Python中函数的参数传递与可变长参数
转自旭东的博客原文 Python中函数的参数传递与可变长参数 Python中传递参数有以下几种类型: (1)像C++一样的默认缺省函数 (2)根据参数名传参数 (3)可变长度参数 示例如下: (1)默 ...
- .NET框架- in ,out, ref , paras使用的代码总结 C#中in,out,ref的作用 C#需知--长度可变参数--Params C#中的 具名参数 和 可选参数 DEMO
C#.net 提供的4个关键字,in,out,ref,paras开发中会经常用到,那么它们如何使用呢? 又有什么区别? 1 in in只用在委托和接口中: 例子: 1 2 3 4 5 6 7 8 9 ...
- python函数中的位置参数、默认参数、关键字参数、可变参数区别
一.位置参数 调用函数时根据函数定义的参数位置来传递参数. #!/usr/bin/env python # coding=utf-8 def print_hello(name, sex): sex_d ...
- Python中为什么不能用可变对象作为默认参数的值
def func(numbers = [], num=1): numbers.append(num) for number in numbers: print(number) func() >& ...
- Python中的*可变参数与**关键字参数
1.定义了一个需要两个参数的函数 def print_str(first, second): print first print second if __name__ == "__main_ ...
- Java中不定项参数(可变参数)的使用
Java1.5增加了新特性:可变参数:适用于参数个数不确定,类型确定的情况,java把可变参数当做数组处理. 注意事项: 1)不定项参数必须放在参数列表最后一个. 2)不定项参数只能有一个(多 ...
- c#基础之长度可变类型相同的参数列表
为了简化编码,c#提供了一个特殊的关键字params,允许在调用方法时提供数量可变的实参,而不是由方法实现固定好的形参数量.先看代码吧. using System; using System.Linq ...
随机推荐
- 【HZERO】分支管理
分支管理 分支类型 feature-[任务编号]-简单描述: 任务开发分支,针对迭代子任务建立的开发分支 bugfix :修复分支,用于缺陷修复. develop:开发分支,所有开发人员都可以提交代码 ...
- Python报错:TypeError: 'dict_keys' object does not support indexing(机器学习实战treePlotter代码)解决方案
错误信息: 学习<机器学习实战>这本书时,按照书上的代码运行,产生了错误,但是在代码中没有错误提示,产生错误的代码如下: firstStr = myTree.keys()[0] print ...
- 【题解】Qin Shi Huang's National Road System HDU - 4081 ⭐⭐⭐⭐ 【次小生成树】
During the Warring States Period of ancient China(476 BC to 221 BC), there were seven kingdoms in Ch ...
- Java 中初始化 List 的五种方法
1.构造 List 后使用 List.add 初始化 1 List<String> stringList = new LinkedList<>(); 2 stringList. ...
- nginx安装 没有网络且缺少基础包的环境下
一.安装 [root@oracle ~]# cd /etc/yum.repos.d/ [root@oracle yum.repos.d]# rm -rf * [root@oracle yum.repo ...
- Qt开发-共享内存使用范例,配合开发者密钥使用后台调试程序或者进入调试模式
共享内存 就之前不是开发了一个Leventure_DeveloperKey用以调试程序嘛,在这里简单聊一下调试模式的方案. 这里的调试分为了两种,一种是调试模式,一种是开发者模式.需要这两种模式的原因 ...
- 使用 Docker 安装 MongoDB 数据库
by emanjusaka from https://www.emanjusaka.top/2024/01/docker-create-mongo-db 彼岸花开可奈何 本文欢迎分享与聚合,全文转载请 ...
- 问题--链表指针传参,修改next指针只传值
1.问题--链表指针传参,修改next指针只传值 Link_creat_head(&head, p_new);//将新节点加入链表 在这当中head头指针传的是地址,而p_new传的是值,这二 ...
- Java-Enum常量特定方法
OnJava8-Enum-常量特定方法 用枚举实现责任链模式 责任链(Chain Of Responsibility)设计模式先创建了一批用于解决目标问题的不同方法,然后将它们连成一条"链& ...
- 【转帖】【性能提升神器】STRAIGHT_JOIN
今天给大家下另一个性能提升神器-STRAIGHT_JOIN,在数据量大的联表查询中灵活运用的话,能大大缩短查询时间. 首先来解释下STRAIGHT_JOIN到底是用做什么的: STRAIGHT_JOI ...