深入C语言可变参数(va_arg,va_list,va_start,va_end,_INTSIZEOF)

转载 2017年05月10日 17:28:56

一、什么是可变参数

         在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剖析

 

  • 内存对齐  #define _INTSIZEOF(n) ((sizeof(n)+sizeof(int)-1)&~(sizeof(int) - 1) ) 
                 具体请参考:http://blog.csdn.net/swell624/article/details/3210779
  • 可变参数用到以下宏函数  
        <1>原型: 
    void va_start(va_list arg_ptr,prev_param);

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

              返回值:无

        <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.cnblogs.com/haoyuanyuan/p/3221463.html

       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

C/C++中va_list,va_start,va_arg,va_end的使用和原理

参考: http://www.cnblogs.com/dongsheng/p/4001555.html http://www.cnblogs.com/yongssu/p/4677556.html...
  • piaoxuezhong
  • piaoxuezhong
  • 2017年03月06日 13:05
  • 942

C语言可变参数函数详解 va_list va_arg va_start宏

可变参数函数: 参数数目是可变的。比如 C 语言的 printf 系列的(格式化输入输出等)函数,都是参数可变的,形式是形参是三个点。 如printf 函数的声明: int printf ( c...
  • geqiandebei
  • geqiandebei
  • 2015年01月05日 15:36
  • 1437

va_list、va_start、va_arg、va_end宏的使用(转)

当你的函数的参数个数不确定时,就可以使用上述宏进行动态处理,这无疑为你的程序增加了灵活性。Example:CString AppendString(CString str1,...)//一个连接字符串...
  • tuwen
  • tuwen
  • 2008年03月21日 14:18
  • 2355

va_list、va_start、va_arg、va_end等利用宏定义实现可变参数

#include //#include //#include //int sum(int n, ...); typedef void* va_list; #define _INTSIZEOF...
  • z251941074
  • z251941074
  • 2015年01月31日 21:13
  • 1386

va_list是一个宏,由va_start和va_end界定,一时难说清,详细见《Windows32程序设计》Unicode部分

va_list是一个宏,由va_start和va_end界定,一时难说清,详细见《Windows32程序设计》Unicode部分va_list   structure       Used   to ...
  • wenzuodan
  • wenzuodan
  • 2008年01月31日 15:50
  • 4069

C语言中可变参数va_list/va_start/value_arg/va_end的理解

va_list/va_start/va_arg/va_end这几个宏,都是用于函数的可变参数的。 我们来看看在vs2008中,它们是怎么定义的: 1: ///stdarg.h 2: ...
  • zx642288904
  • zx642288904
  • 2017年03月07日 16:20
  • 400

ios 可变参数(va_list,va_start,va_end)

原文地址:ios 可变参数(va_list,va_start,va_end) 例如:UIAlertView的init方法中的otherButtonTitles:(NSString *)oth...
  • u012894479
  • u012894479
  • 2015年06月23日 15:18
  • 1732

C/C++(6)va_list、va_start、va_arg、va_end实现可变参数列表函数

经典的可变参数表printf函数的实现原理:在C/C++中,对函数参数的扫描是从后向前的。 函数参数是通过压入堆栈的方式来给函数传参数的,最先压入的参数最后出来,而栈是从内存的高地址向低地址存放的,...
  • qjpxAAA
  • qjpxAAA
  • 2016年10月24日 17:40
  • 410

C语言可变参数,va_arg、va_list,va_start,va_end,_INTSIZEOF

http://www.cnblogs.com/wucg/archive/2012/02/26/2368354.html   http://blog.sina.com.cn/s/blog_62050...
  • helinsen
  • helinsen
  • 2013年04月07日 14:44
  • 353

va_start、va_arg、va_end用法

1:当无法列出传递函数的所有实参的类型和数目时,可用省略号指定参数表void foo(...);void foo(parm_list,...);2:函数参数的传递原理函数参数是以数据结构:栈的形式存取...
  • lanyzh0909
  • lanyzh0909
  • 2010年06月27日 22:22
  • 10805
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:深入C语言可变参数(va_arg,va_list,va_start,va_end,_INTSIZEOF)
举报原因:
原因补充:

(最多只允许输入30个字)