转自:http://www.cnblogs.com/chinazhangjie/archive/2012/08/18/2645475.html

一、基础部分

1.1 什么是可变长参数

可变长参数:顾名思义,就是函数的参数长度(数量)是可变的。比如 C 语言的 printf 系列的(格式化输入输出等)函数,都是参数可变的。下面是 printf 函数的声明:

int printf ( const char * format, ... );

可变参数函数声明方式都是类似的。

1.2 如何实现

C语言可变参数通过三个宏(va_start、va_end、va_arg)和一个类型(va_list)实现的,

void va_start ( va_list ap, paramN );
参数:
ap: 可变参数列表地址 
paramN: 确定的参数
功能:初始化可变参数列表(把函数在 paramN 之后的参数地址放到 ap 中)。

void va_end ( va_list ap );
功能:关闭初始化列表(将 ap 置空)。

type va_arg ( va_list ap, type );
功能:返回下一个参数的值。

va_list :存储参数的类型信息。

好了,综合上面3个宏和一个类型可以猜出如何实现C语言可变长参数函数:用 va_start 获取参数列表(的地址)存储到 ap 中,用 va_arg 逐个获取值,最后用 va_arg 将 ap 置空。

1.3 举例

  1. /* 作者:独酌逸醉
  2. * 时间:2012.08.18
  3. * 功能:用C语言实现变长参数小例:求和
  4. * IDE: Microsoft Visual Studio 2010
  5. */
  6.  
  7. #include <stdio.h>
  8. #include <stdarg.h>
  9.  
  10. #define END -1
  11.  
  12. int va_sum (int first_num, ...)
  13. {
  14. // (1) 定义参数列表
  15. va_list ap;
  16. // (2) 初始化参数列表
  17. va_start(ap, first_num);
  18.  
  19. int result = first_num;
  20. int temp = ;
  21. // 获取参数值
  22. while ((temp = va_arg(ap, int)) != END)
  23. {
  24. result += temp;
  25. }
  26.  
  27. // 关闭参数列表
  28. va_end(ap);
  29.  
  30. return result;
  31. }
  32.  
  33. int main ()
  34. {
  35. int sum_val = va_sum(, , , , , END);
  36. printf ("%d", sum_val);
  37. return ;
  38. }

1.4 使用注意事项

  1. 宏定义在 stdarg.h 中,所以使用时,不要忘了添加头文件。
  2. 设定一个参数结束标志(cplusplus 上说,va_arg 并不能确定哪个参数是最后一个参数)。
  3. 类型的匹配
  4. 期待您的补充……

二、深入原理

“源码面前,一览无遗”!

以下源码,来自“..\Microsoft Visual Studio 10.0\VC\include”

  1. // stdarg.h
  2. #define va_start _crt_va_start
  3. #define va_arg _crt_va_arg
  4. #define va_end _crt_va_end
  5. // vadefs.h
  6. typedef char *  va_list;
  7. #define _crt_va_start(ap,v)  ( ap = (va_list)_ADDRESSOF(v) + _INTSIZEOF(v) )
  8. #define _crt_va_arg(ap,t)    ( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) )
  9. #define _crt_va_end(ap)      ( ap = (va_list)0 )
  10. #define _ADDRESSOF(v)   ( &(v) )
  11. #define _INTSIZEOF(n)   ( (sizeof(n) + sizeof(int) - 1) & ~(sizeof(int) - 1) )

除了 _INTSIZEOF 之外,其他都很好理解,举个例子吧:

  1. /* 作者:独酌逸醉
  2.  * 时间:2012.08.18
  3.  * 功能:测试 _INTSIZEOF 宏
  4.  * IDE:  CodeBlocks 10.05
  5.  */
  6.  
  7. #include <stdio.h>
  8. #include <stdarg.h>
  9.  
  10.  
  11. int main ()
  12. {
  13.     int i = ;
  14.     float f = 0.0;
  15.     printf("_INTSIZEOF(i) = %d\n", (int)(_INTSIZEOF(i)));
  16.     printf("_INTSIZEOF(f) = %d\n", (int)(_INTSIZEOF(f)));
  17.     printf("_INTSIZEOF(\"Hello,world\") = %d\n", (int)(_INTSIZEOF("Hello,world")));
  18.     printf("sizeof(\"Hello,world\") = %d\n", sizeof("Hello,world") );
  19.     return ;
  20. }

输出结果:

  1. _INTSIZEOF(i) =
  2. _INTSIZEOF(f) =
  3. _INTSIZEOF("Hello,world") =
  4. sizeof("Hello,world") =

既然 sizeof 和 _INTSIZEOF 值一样,为什么不直接用 sizeof 呢?干嘛要写的那么复杂?答案是为了字节对齐(无论32位还是64位机器,sizeof(int)永远代表机器的位数,明白了吧!^_^)

现在再去看变长参数的实现:其实就是把参数在栈中的地址记录到 ap 中(通过一个确定参数 paramN 确定地址),然后逐个读取值。

此时是否有一种豁然开朗的感觉?至少明白了许多,也清楚了很多。


三、知识扩展

可能大家也猜到了,我扩展要扩展什么了?!^_^

简单介绍两种函数调用约定

__stdcall (C++默认)

  1. 参数从右向左压入堆栈
  2. 函数被调用者修改堆栈
  3. 函数名(在编译器这个层次)自动加前导的下划线,后面紧跟一个@符号,其后紧跟着参数的尺寸

__cdecl (C语言默认)

  1. 参数从右向左压入堆栈
  2. 参数由调用者清楚,手动清栈,被调用函数不会要求调用者传递多少参数,调用者传递过多或者过少的参数,甚至完全不同的参数都不会产生编译阶段的错误。

那么,变参函数的调用方式为(也只能是):__cdecl 。

本来打算多写一点扩展的,又担心会文不符题,所以感兴趣的朋友可以去看参考资料中的文章,有一些介绍的很详细。

参考资料

  1. http://www.cplusplus.com/reference/clibrary/cstdarg/va_start/
  2. http://www.cplusplus.com/reference/clibrary/cstdarg/va_end/
  3. http://www.cplusplus.com/reference/clibrary/cstdarg/va_list/
  4. http://www.cplusplus.com/reference/clibrary/cstdarg/va_arg/
  5. http://51hired.com/questions/13278?sort=oldest
  6. http://www.cnblogs.com/diyunpeng/archive/2010/01/09/1643160.html
  7. http://blog.csdn.net/huanjieshuijing/article/details/5822942
  8. http://baike.baidu.com/view/1280676.htm

[转]深度探索C语言函数可变长参数的更多相关文章

  1. Noah的学习笔记之Python篇:函数“可变长参数”

    Noah的学习笔记之Python篇: 1.装饰器 2.函数“可变长参数” 3.命令行解析 注:本文全原创,作者:Noah Zhang  (http://www.cnblogs.com/noahzn/) ...

  2. 0521Day03命名规范 Data函数 可变长参数 枚举类型

    [重点] 命名规范 枚举类型 Date函数 可变长参数 pirnt,println 命名规范 1. 驼峰命名法:main,username,setUsername 用于变量.方法的命名 2. Pasc ...

  3. C++ 系列:函数可变长参数

    一.基础部分 1.1 什么是可变长参数 可变长参数:顾名思义,就是函数的参数长度(数量)是可变的.比如 C 语言的 printf 系列的(格式化输入输出等)函数,都是参数可变的.下面是 printf ...

  4. python 函数可变长参数

    python中的可变长参数有两种: 一种是非关键字参数(*元组),另一种是关键字参数(**字典) 非关键字可变长参数: """ 非关键字可变参数,一个星号作为元组传入函数 ...

  5. C语言之可变长参数格式化

    概述 本文演示环境: win10 + Vs2015 可变长参数格式化 两个概念: 1. 参数长度不定, 2. 参数格式化. 使用函数 vsnprintf 结合 va_list. 源码 写好了函数, 照 ...

  6. 关于C中可变长参数

    前言 可变长参数指函数的参数个数在调用时才能确定的函数参数.基本上各种语言都支持可变长参数,在特定情形下,可变长参数使用起来非常方便.c语言中函数可变长参数使用“...”来表示,同时可变长参数只能位于 ...

  7. C语言开发具有可变长参数的函数的方法

    学习交流可加 微信读者交流①群 (添加微信:coderAllen) 程序员技术QQ交流①群:736386324 --- 前提:ANSI C 为了提高可移植性, 通过头文件stdarg.h提供了一组方便 ...

  8. go实例—函数或方法的可变长参数

    支持可变长参数列表的函数可以支持任意个传入参数,比如fmt.Println函数就是一个支持可变长参数列表的函数. 需要注意的是,可变长参数应该是函数定义的最右边的参数,即最后一个参数 package ...

  9. python基础语法5 函数定义,可变长参数

    函数 1.什么是函数 函数就是一种工具. 可以重复调用 2.为什么要用函数 1.防止代码冗(rong)余 2.代码的可读性差 3.怎么用函数 1.定义函数-->制造工具 2.调用函数--> ...

随机推荐

  1. 通过pull解析器操作安卓的xml

    通过pull解析器操作安卓的xml 例子定义了一个javabean用于存放上面解析出来的xml内容, 这个javabean为Person,代码请见本页下面备注: =================== ...

  2. Linux用户及用户组管理

    Linux是个优秀的多用户多任务操作系统. 掌握Linux的用户/用户组管理是基本及必备技能之一. 简单做下总结. 无论采用图形界面的用户管理设置,还是终端的管理方式,最终目的都是对系统的用户/用户组 ...

  3. android入门——Activity(1)

    结构图 mainfests文件夹下面有一个AndroidMainfest.xml文件,类似web开发中的web.xml文件负责整个项目的配置,每当我们新建一个activity,都要在这个文件中进行配置 ...

  4. Deep Learning(深度学习)学习笔记整理系列之(四)

    Deep Learning(深度学习)学习笔记整理系列 zouxy09@qq.com http://blog.csdn.net/zouxy09 作者:Zouxy version 1.0 2013-04 ...

  5. CSS截取字符串

    /*溢出的字...处理*/ .updatecsssubstring { text-overflow: ellipsis; -o-text-overflow: ellipsis; white-space ...

  6. 3种方式实现可滑动的Tab

    1. 第一种,使用 TabHost + ViewPager 实现 该方法会有一个Bug,当设置tabHost.setCurrentTab()为0时,ViewPager不显示(准确的说是加载),只有点击 ...

  7. Linux 组与用户

    组: 添加: groupadd groupName -g groupID  --> groupadd dba -g 502 删除: groupdel  groupName             ...

  8. 转: git常用命令

    # git配置 #---------------------------------------------- #配置用户名和邮箱: $ git config --global user.name & ...

  9. 把WinXP装进内存 性能飚升秒杀固态硬盘

    现在用户新配置的电脑,内存很少有小于2GB的,配置4GB内存的朋友也有不少.容量如此大的内存,我们在使用电脑的日常操作中绝对用不完.而目前制约系统性能最大的瓶颈就是硬盘的传输速度,所以,这里教你怎么把 ...

  10. Speex manul中文版

    Speex manul中文版   在VOIP的音频算法中,回音处理已经成为一个关系通话质量的主要问题. 回声的产生在IP网络主要有两种:1.声学回声2.电路回声 声学回声主要又分成以下几种:a ) 直 ...