可变参数学习笔记

laomai 2004-11-24 09:44:54
最近应CSDN的邀请,C/C++值班室的几位兄弟为C++电子杂志写了一些文章,现将我的稿件预先刊发在论坛上,请兄弟们批评指正。也欢迎大家为CSDN c/c++电子杂志专刊投稿。杂志详情请见http://emag.csdn.net/
可变参数学习笔记
作者:laomai(原创,转载时请注明来自CSDN 的论坛及c/c++电子杂志)
前言:
本文在很大程度上改编自网友kevintz的“C语言中可变参数的用法”一文,在行文之前先向这位前辈表示真诚的敬意和感谢。
一、什么是可变参数
我们在C语言编程中有时会遇到一些参数个数可变的函数,例如printf()函数,其函数原型为:
int printf( const char* format, ...);
它除了有一个参数format固定以外,后面跟的参数的个数和类型是可变的(用三个点“…”做参数占位符),实际调用时可以有以下的形式:
printf("%d",i);
printf("%s",s);
printf("the number is %d ,string is:%s", i, s);
以上这些东西已为大家所熟悉。但是究竟如何写可变参数的C函数以及这些可变参数的函数编译器是如何实现,这个问题却一直困扰了我好久。本文就这个问题进行一些探讨,希望能对大家有些帮助.

二、写一个简单的可变参数的C函数
先看例子程序。该函数至少有一个整数参数,其后是占位符…,表示后面参数的个数不定. 在这个例子里,所有的输入参数必须都是整数,函数的功能是打印所有参数的值.
函数代码如下:
//示例代码1:可变参数函数的使用
#include "stdio.h"
#include "stdarg.h"
void simple_va_fun(int start, ...)
{
va_list arg_ptr;
int nArgValue =start;
int nArgCout=0; //可变参数的数目
va_start(arg_ptr,start); //以固定参数的地址为起点确定变参的内存起始地址。
do
{
++nArgCout;
printf("the %d th arg: %d\n",nArgCout,nArgValue); //输出各参数的值
nArgValue = va_arg(arg_ptr,int); //得到下一个可变参数的值
} while(nArgValue != -1);
return;
}
int main(int argc, char* argv[])
{
simple_va_fun(100,-1);
simple_va_fun(100,200,-1);
return 0;
}

从这个函数的实现可以看到,我们使用可变参数应该有以下步骤:
⑴在程序中将用到以下这些宏:
void va_start( va_list arg_ptr, prev_param );
type va_arg( va_list arg_ptr, type );
void va_end( va_list arg_ptr );
va在这里是variable-argument(可变参数)的意思.
这些宏定义在stdarg.h中,所以用到可变参数的程序应该包含这个头文件.
⑵函数里首先定义一个va_list型的变量,这里是arg_ptr,这个变量是指向参数地址的指针.因为得到参数的地址之后,再结合参数的类型,才能得到参数的值。
⑶然后用va_start宏初始化⑵中定义的变量arg_ptr,这个宏的第二个参数是可变参数列表的前一个参数,也就是最后一个固定参数。
⑷然后依次用va_arg宏使arg_ptr返回可变参数的地址,得到这个地址之后,结合参数的类型,就可以得到参数的值。然后进行输出。
⑸设定结束条件,这里的条件就是判断参数值是否为-1。注意被调的函数在调用时是不知道可变参数的正确数目的,程序员必须自己在代码中指明结束条件。至于为什么它不会知道参数的数目,读者在看完下面这几个宏的内部实现机制后,自然就会明白。
...全文
3788 53 打赏 收藏 转发到动态 举报
写回复
用AI写文章
53 条回复
切换为时间正序
请发表友善的回复…
发表回复
zhousqy 2005-06-08
  • 打赏
  • 举报
回复
好贴,收藏.
carroty 2005-03-07
  • 打赏
  • 举报
回复
mark
BINARYKNIGHT 2005-03-05
  • 打赏
  • 举报
回复
mark
lemon520 2005-03-04
  • 打赏
  • 举报
回复
mark
eduhf_123 2004-12-28
  • 打赏
  • 举报
回复
收藏
b46 2004-12-28
  • 打赏
  • 举报
回复
UP
ilovevc 2004-12-28
  • 打赏
  • 举报
回复
很好的文章,不过对
#define va_end(ap) ( ap = (va_list)0 )
它并不仅仅是一个好的习惯,还有其他的用途。

例如可能有的实现va_start修改了栈指针,那么va_end就提供了一个机会恢复该指针。当然具体到某些实现,可能va_end仅仅是一个dummy实现就可以了。

bill_li 2004-12-28
  • 打赏
  • 举报
回复
up
lqgoal 2004-12-28
  • 打赏
  • 举报
回复
Deitel父子编写的C程序设计教程——高级话题部分 介绍了可变参数的函数,有实例
dongpy 2004-12-27
  • 打赏
  • 举报
回复
学习
beyondtkl 2004-12-27
  • 打赏
  • 举报
回复
MK
myc 2004-12-04
  • 打赏
  • 举报
回复
mark
superpig2k 2004-12-04
  • 打赏
  • 举报
回复
mark先
cxc014 2004-12-04
  • 打赏
  • 举报
回复
占个位先
williamVII 2004-12-04
  • 打赏
  • 举报
回复
too strong!
superxxm 2004-12-04
  • 打赏
  • 举报
回复
好啊!
stevens2009 2004-12-04
  • 打赏
  • 举报
回复
mark
DuoFG 2004-12-03
  • 打赏
  • 举报
回复
慢慢看
hcj2002 2004-12-03
  • 打赏
  • 举报
回复
收藏:(
dongnan 2004-12-03
  • 打赏
  • 举报
回复
支持
加载更多回复(33)

69,373

社区成员

发帖
与我相关
我的任务
社区描述
C语言相关问题讨论
社区管理员
  • C语言
  • 花神庙码农
  • 架构师李肯
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
暂无公告

试试用AI创作助手写篇文章吧