stdarg.h中三个宏va_start ,va_arg和va_end

可以到stgarg.h文件中查看定义发现如下内容。

/* Define __gnuc_va_list. */

#ifndef __GNUC_VA_LIST
#define __GNUC_VA_LIST
typedef __builtin_va_list __gnuc_va_list;
#endif

#define va_start(v,l) __builtin_va_start(v,l)
#define va_end(v) __builtin_va_end(v)
#define va_arg(v,l) __builtin_va_arg(v,l)
#if !defined(__STRICT_ANSI__) || __STDC_VERSION__ + 0 >= 199900L || defined(__GXX_EXPERIMENTAL_CXX0X__)
#define va_copy(d,s) __builtin_va_copy(d,s)
#endif
#define __va_copy(d,s) __builtin_va_copy(d,s)

 

一、什么是可变参数

在C语言编程中有时会遇到一些参数个数可变的函数,例如printf(),scanf()函数,其函数原型为:
int printf(const char* format,…),int scanf(const char *format,…);它除了有一个参数format固定以外,后面跟着的参数的个数和类型是可变的(用三个点“…”做参数占位符),实际调用时可以有以下的形式:
printf(“%d”,i);  printf(“%d,%c”,i,j);

二、C语言—简单的可变参数例子(实现思想)

        1、整型数据的输出

#include <stdio.h>
#include <stdarg.h>
void ar_cnt(int cnt,...);
void ar_cst(char const *s,...);
int main(int argc, char* argv[])
{
    int in_size =_INTSIZEOF(int);
    printf("int_size=%d\n",in_size);
    ar_cnt(5,1,2,3,4);
    return 0; 
}
void ar_cnt(int cnt,...)
{
    int value1=0;
    int i=0;
    int arg_cnt = cnt;
    va_list arg_ptr;
    va_start(arg_ptr,cnt);
    for(i=0;i<cnt;i++)
    {
        value1=va_arg(arg_ptr,int);
        printf("posation %d=%d\n",value1,i+1);
    }
    va_end(arg_ptr);
}

运行结果:

image

 

2、字符串的输出

#include <stdio.h>
#include <stdarg.h>

void PrintLines(char *first,...)
{
    char *str;
    va_list v1;
    str = first;
    va_start(v1,first);
    do 
    {
        printf("%s\n",str);
        str=va_arg(v1,char*);
    } while (str != NULL );
    va_end(v1);
}

int main(int argc, char* argv[])
{
    PrintLines("First","Second","Third","Fourth",NULL);
    return 0;
}

运行结果:

image

           3、找出最大数

#include <stdio.h>
#include <stdarg.h>
int FindMax(int amount,...)
{
    int i,val,great;
    va_list v1;
    va_start(v1,amount);
    great=va_arg(v1,int);
    for(i=1;i<amount;i++)
    {
        val=va_arg(v1,int);
        great=(great>val)?great:val;
    }
    va_end(v1);
    return great;
}

int main(int argc, char* argv[])
{
    int max=FindMax(5,100,20,456,102,4,300);
    printf("The Max one is %d\n",max);
    return 0;
}

运行结果:

image

三、对va_arg,va_list,va_start,va_end,_INTSIZEOF剖析

 

功能:以固定参数的地址为起点确定变参的内存起始地址,获取第一个参数的首地址

返回值:无

<2>原型:va_list 类型的变量,va_list arg_ptr ,这个变量是指向参数地址的指针,因为得到参数的地址之后,再结合参数的类型,才能得到参数的值。

<3>原型:type va_arg(va_list arg_ptr,type);

功能:获取下一个参数的地址

返回值:根据传入参数类型决定返回值类型

<4>原型:void  va_end(va_list arg_ptr);

功能:将arg_ptr指针置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。注意被调的函数在调用时是不知道可变参数的正确数目的,程序员必须自己在代码中指明结束条件。至于为什么它不会知道参数的数目,读者在看完下面这几个宏的内部实现机制后,自然就会明白。

三、参考文献:
http://www.chineselinuxuniversity.net/articles/26262.shtml

http://www.cnblogs.com/wangyonghui/archive/2010/07/12/1776068.html

http://www.2cto.com/kf/201204/129038.html

 

 

 

发表评论

电子邮件地址不会被公开。 必填项已用 * 标注

您可以使用这些 HTML 标签和属性: <a href="" title=""> <abbr title=""> <acronym title=""> <b> <blockquote cite=""> <cite> <code> <del datetime=""> <em> <i> <q cite=""> <strike> <strong>