c(++)变长参数之整形(非字符串类型类似)
0、序言
变长参数,接触的第一个可变长参数函数是 printf , 然后是 scanf 。他们的原型如下:
printf:
_Check_return_opt_
_CRT_STDIO_INLINE int __CRTDECL printf(
_In_z_ _Printf_format_string_ char const* const _Format,
...)
scanf
_CRT_STDIO_INLINE int __CRTDECL scanf(
_In_z_ _Scanf_format_string_ char const* const _Format,
...)
本文演示的是数据类型是 int .
1、使用
A、头文件
// 使用va_start需要的头文件
#include <stdarg.h>
B、使用必须使用的4个宏,原型就不贴出来了,大家更关心的是怎么使用。:
va_list :保存可变长参数
va_start:构建一个参数集合
va_arg:获取集合中的参数
va_end:释放集合
C、和普通参数使用相似,第一个参数必须要指定为类型,后面 写三个 点。例如:
vodi add(const int param, ...);
2、一个完整的例子
演示环境: VS2015 up3
这里,定义了一个函数,函数用来求和,参数类型为整形,参数为可变长参数
1 #include <iostream>
2
3 // 使用va_start需要的头文件
4 #include <stdarg.h>
5
6 // 不定长参数求和
7 void va_sum(const int first_param, ...)
8 {
9
10 va_list ap;
11
12 va_start(ap, first_param);
13
14 // 保存求和的结果, 演示不定长参数用法,暂时不考虑越界
15 int sum = 0;
16
17 // 中间每一项
18 int tmp = 0;
19
20 // -1 表示 集合的结束
21 for (int i = 0; -1 != tmp ; i++)
22 {
23 // 显示第i个参数
24 std::cout << "i=" << i << ", tmp=" << tmp << "\n";
25 // 保存结果
26 sum += tmp;
27 // 找到下一个参数
28 tmp = va_arg(ap, int);
29
30 }
31
32 // 释放空间
33 va_end(ap);
34
35 // 输出结果
36 std::cout << "sum = " << sum << "\n";
37 }
38
39 // 入口函数
40 int main(int argc, char *argv[])
41 {
42 // -1 表示可变长参数的结束
43 va_sum(1, 2, 3, 4, -1);
44
45 system("pause");
46 return 0;
47 }
这段代码,大家先看下, 预测下 sum 的输出结果。 下面将有正确答案。
我第一次作答的答案: 函数 va_sum 用来求和的, main函数传入了参数: 1. 2. 3. 4. -1. 所以, va_sum中的sum=1 + 2 + 3 + 4 = 10。 sum将输出 10。
正确答案是:
sum 怎么不是10 ? 怎么i= 0,输出的不是第一个参数1, 而是 2?
踩坑:检查发现, sum 少加了 参数 1。 阅读上面的代码,不难发现, 当 第一次 调用 va_arg 宏时, 此时得到的是集合的第1个参数,而不是调用传入的参数1. 1 是第一个参数。
具体,咱们看下 va_start, va_end, va_arg的宏定义(来自Vs2015up3中vadefs.h文件的定义)
1 #define __crt_va_start_a(ap, v) ((void)(ap = (va_list)_ADDRESSOF(v) + _INTSIZEOF(v)))
2 #define __crt_va_arg(ap, t) (*(t*)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)))
3 #define __crt_va_end(ap) ((void)(ap = (va_list)0))
明白了。
怎么改实现求和? sum 加上 第一个参数 :first_param 即可。va_sum 函数的for循环前增加一行代码:
// 新增加
sum += first_param;
va_sum函数完整定义:
1 // 不定长参数求和
2 void va_sum(const int first_param, ...)
3 {
4
5 va_list ap;
6
7 va_start(ap, first_param);
8
9 // 保存求和的结果, 演示不定长参数用法,暂时不考虑越界
10 int sum = 0;
11
12 // 中间每一项
13 int tmp = 0;
14
15 // 新增加
16 sum += first_param;
17
18 // -1 表示 集合的结束
19 for (int i = 0; -1 != tmp ; i++)
20 {
21 // 显示第i个参数
22 std::cout << "i=" << i << ", tmp=" << tmp << "\n";
23 // 保存结果
24 sum += tmp;
25 // 找到下一个参数
26 tmp = va_arg(ap, int);
27
28 }
29
30 // 释放空间
31 va_end(ap);
32
33 // 输出结果
34 std::cout << "sum = " << sum << "\n";
35 }
输出结果:
3、总结
A、第一次调用 va_arg 指向的是集合的第一个元素,而不是调用时传入的第一个参数。
B、va_list 保存的时 函数参数 三个点【...】的参数,调用时传入的第一个参数保存在函数的第一个参数中。
C、 va_start 与 va_end 需要配对使用。
D、需要指定结束符。 这里 -1 为结束符。
c(++)变长参数之整形(非字符串类型类似)的更多相关文章
- C++中的变长参数
新参与的项目中,为了使用共享内存和自定义内存池,我们自己定义了MemNew函数,且在函数内部对于非pod类型自动执行构造函数.在需要的地方调用自定义的MemNew函数.这样就带来一个问题,使用stl的 ...
- Java基础12-工具类;变长参数;IO
作业解析 取出整数的16进制表示形式 \u00ff /** * int2hex * */ public static String int2hex(int i) { String str = &quo ...
- 《OOC》笔记(3)——C语言变长参数va_list的用法
<OOC>笔记(3)——C语言变长参数va_list的用法 C语言中赫赫有名的printf函数,能够接受的参数数目不固定,这就是变长参数.C#里也有params这个关键字用来实现变长参数. ...
- (一)预定义宏、__func__、_Pragma、变长参数宏定义以及__VA_ARGS__
作为第一篇,首先要说一下C++11与C99的兼容性. C++11将 对以下这些C99特性的支持 都纳入新标准中: 1) C99中的预定义宏 2) __func__预定义标识符 3) _Pragma操作 ...
- c++11变长参数函数模板
By francis_hao Mar 25,2018 一个最简单的实例大概是这个样子: #include <iostream>using namespace std; /*变长参 ...
- Scala 变长参数
如果Scala定义变长参数 def sum(i Int*), 那么调用sum时,可以直接输入sum(1,2,3,4,5) 但是不可以sum(1 to 5) 必须要将1 to 5 强制为seq sum( ...
- C++11变长参数模板
[C++11变长参数模板] C++03只有固定模板参数.C++11 加入新的表示法,允许任意个数.任意类别的模板参数,不必在定义时将参数的个数固定. 实参的个数也可以是 0,所以 tuple<& ...
- 【Unix环境高级编程】编写变长参数函数
文件的格式输入输出函数都支持变长参数.定义时,变长参数列表通过省略号'...'表示, 因此函数定义格式为: type 函数名(parm1, parm2,parmN,...); Unix的变长参数通过v ...
- java常量和变量的定义规则,变长参数的使用
首先是定义的一般规则,类名首字母全部大写,常量全部大写用下划线分隔,变量用驼峰形式.注意使用long赋值用L时不能写小写的L要写大写的,不然会和数字“1”傻傻分不清. 下面是举例: public cl ...
随机推荐
- 【机器学习与R语言】3-概率学习朴素贝叶斯(NB)
目录 1.理解朴素贝叶斯 1)基本概念 2)朴素贝叶斯算法 2.朴素贝斯分类应用 1)收集数据 2)探索和准备数据 3)训练模型 4)评估模型性能 5)提升模型性能 1.理解朴素贝叶斯 1)基本概念 ...
- 解决sourceforge下载文件慢的方法
Sourceforge是一些开源软件经常用到的网站,然而国内的网站一直不稳定,如今是可以访问,但是一直无法下载,或者是下载速度慢,导致下载中断 镜像源:http://sourceforge.mirro ...
- CentOS6.9 内核升级详解
内核进行的是应用软件和计算机硬件的交互工作在计算机科学中,内核(英语:kernel)又称核心,是一个计算机程序,用来管理软件发出的数据I/O(输入与输出)要求,将这些要求转译为数据处理的指令,交由中央 ...
- Python与Perl的相似与差别
Python version 3.7版本 00.命令行交互 命令行交互 Perl Python perl -e <Perl代码> #Unix/Linux/Windows/DOS 直 ...
- LeetCode数组中重复的数字
LeetCode 数组中重复的数字 题目描述 在一个长度为 n 的数组 nums 里的所有数字都在 0~n-1 的范围内.数组中某些数字是重复的,但不知道有几个数字重复了,也不知道每个数字重复了几次. ...
- Hadoop的HA机制浅析
Zookeeper在Hadoop的HA中的应用 非HA的弊端: HDFS集群的分布式存储是靠namenode节点(namenode负责响应客户端请求)来实现.在非HA集群中一旦namenode宕机,虽 ...
- 30个类手写Spring核心原理之环境准备(1)
本文节选自<Spring 5核心原理> 1 IDEA集成Lombok插件 1.1 安装插件 IntelliJ IDEA是一款非常优秀的集成开发工具,功能强大,而且插件众多.Lombok是开 ...
- JAVA中的六种日期类型使用
基本的6种日期类 /** * 六种时间类型的类 * 数据库格式的时间三种格式 */ java.util.Date date = new java.util.Date();//年与日时分秒 //数据库的 ...
- 利用Lombok编写优雅的spring依赖注入代码,去掉繁人的@Autowired
大家平时使用spring依赖注入,都是怎么写的? @Servicepublic class OrderService {@Autowiredprivate UserService userServic ...
- Python @函数装饰器及用法(超级详细)
函数装饰器的工作原理是怎样的呢?假设用 funA() 函数装饰器去装饰 funB() 函数,如下所示: #funA 作为装饰器函数 def funA(fn): #... fn() # 执行传入的fn参 ...