快捷搜索:  as  test  1111  test aNd 8=8  test++aNd+8=8  as++aNd+8=8  as aNd 8=8

和记娱乐和记怡情AG:C语言中可变参数的用法



我们在C说话编程中会碰到一些参数个数可变的函数,例如printf()

这个函数,它的定义是这样的:

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

它除了有一个参数for和记娱乐和记怡情AGmat固定以外,后面跟的参数的个数和类型是可变的,例如我们可以有以下不合的调用措施:

printf("%d",i);

printf("%s",s);

printf("the  number  is  %d  ,string  is:%s",  i,  s);

究竟若何写可变参数的C函数以及这些可变参数的函数编译器是若何实现的呢?本文就这个问题进行一些探究,和记娱乐和记怡情AG盼望能对大年夜家有些赞助.会C++的网友知道这些问题在C++里不存在,由于C++具有多态性.但C++是C的一个超集,以下的技巧也可以用于C++的法度榜样中.限于本人的水平,文中假如有 欠妥之处,请大年夜家斧正.

(一)写一个简单的可变参数的C函数

下面我们来探究若何写一个简单的可变参数的C函数.写可变参数的C函数要在法度榜样顶用到以下这些宏:

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中,以是用到可变参数的法度榜样应该包孕这个 头文件.下面我们写一个简单的可变参数的函数,改函数至少有一个整数参数,第二个参数也是整数,是可选的.函数只是打印这两个参数的值.

void  simple_va_fun(int  i,  ...)

{

va_list  arg_ptr;

int  j=0;

va_start(arg_ptr,  i);

j=va_arg(arg_ptr,  int和记娱乐和记怡情AG);

va_end(arg_ptr);

printf("%d  %dn",  i,  j);

return;

}

我们可以在我们的头文件中这样声明我们的函数:

extern  void  simple_va_fun(int  i,  ...);

我们在法度榜样中可以这样调用:

simple_va_fun(100);

simple_va_fun(100,200);

从这个函数的实现可以看到,我们应用可变参数应该有以下步骤:

1)首先在函数里定义一个va_list型的变量,这里是arg_ptr,这个变量是指向参数和记娱乐和记怡情AG的指针.

2)然后用va_start宏初始化变量arg_ptr,这个宏的第二个参数是第一个可变参数的前一个参数,是一个固定的参数.

3)然后用va_arg返回可变的参数,并赋值给整数j.  va_arg的第二 参数是你要返回的参数的类型,这里是int型.

4)着末用va_end宏停止可变参数的获取.然后你就可以在函数里应用第二个参数了.假如函数有多个可变参数的,依次调用va_arg获取各个参数.

假如我们用下面三种措施调用的话,都是合法的,但结果却不一样:

1)simple_va_fun(100);

结果是:100  -123456789(会变的值)

2)simple_va_fun(100,200);

结果是:100  200

3)simple_va_fun(100,200,300);

结果是:100  200

我们看到第一种调用有差错,第二种调用精确,第三种调用只管结果精确,但和我们函数最初的设计有冲突.下面一节我们探究呈现这些结果的缘故原由和可变参数在编译器中是若何处置惩罚的.

(二)可变参数在编译器中的处置惩罚

我们知道va_start,va_arg,va_end是在stdarg.h中被定义成宏的, 因为1)硬件平台的不合  2)编译器的不合,以是定义的宏也有所不合,下面以VC++中stdarg.h里x86平台的宏定义摘录如下(’’号表示折行):

高地址|-----------------------------|

|函数返回地址  |

|-----------------------------|

|.......  |

|-----------------------------| void  simple_va_fun(int  i,  ...)

{

va_list  arg_ptr;

char  *s=NULL;

va_start(arg_ptr,  i);

s=va_arg(arg_ptr,  char*);

va_end(arg_ptr);

printf("%d  %sn",  i,  s);

return;

}

可变参数为char*型,当我们忘怀用两个参数来调用该函数时,就会呈现和记娱乐和记怡情AGcore  dump(Unix)  或者页面不法的差错(window平台).但也有可能不掉足,但差错却是难以发明,晦气于我们写出高质量的法度榜样.

以下提一下va系列宏的兼容性.

System  V  Unix把va_start定义为只有一个参数的宏:

va_start(va_list  arg_ptr);

而ANSI  C则定义为:

va_start(va_list  arg_ptr,  prev_param);

假如我们要用system  V的定义,应该用vararg.h头文件中所定义的宏,ANSI  C的宏跟system  V的宏是不兼容的,我们一样平常都用ANSI  C,以是用ANSI  C的定义就够了,也便于法度榜样的移植.

小结:

可变参数的函数道理着实很简单,而va系列因此宏定义来定义的,实现跟客栈相关.我们写一个可变函数的C函数时,有利也有弊,以是在不需要的场合,我们无需用到可变参数.假如在C++里,我们应该使用C++的多态性来实现可变参数的功能,只管即便避免用C说话的要领来实现.

免责声明:以上内容源自网络,版权归原作者所有,如有侵犯您的原创版权请告知,我们将尽快删除相关内容。

您可能还会对下面的文章感兴趣: