C language variable arguments

今天看apue的头文件,里面出现类是这样的东东:

void    err_msg(const char *, ...);            /* {App misc_source} */
void    err_dump(const char *, ...) __attribute__((noreturn));
void    err_quit(const char *, ...) __attribute__((noreturn));
void    err_cont(int, const char *, ...);
void    err_exit(int, const char *, ...) __attribute__((noreturn));
void    err_ret(const char *, ...);
void    err_sys(const char *, ...) __attribute__((noreturn));

然后就在网上搜索变长参数问题。

一、基础部分

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 置空。

/* 作者:独酌逸醉
 * 时间:2012.08.18
 * 功能:用C语言实现变长参数小例:求和
 * IDE:  Microsoft Visual Studio 2010 Or GNU C
 */

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

#define END -1

int va_sum (int first_num, ...)
{
    // (1) 定义参数列表
    va_list ap;
    // (2) 初始化参数列表
    va_start(ap, first_num);

    int result = first_num;
    int temp = 0;
    // 获取参数值
    while ((temp = va_arg(ap, int)) != END)
    {
        result += temp;
    }

    // 关闭参数列表
    va_end(ap);

    return result;
}

int main ()
{
    int sum_val = va_sum(1, 2, 3, 4, 5, END);
    printf ("%d", sum_val);
    return 0;
}

1.4 使用注意事项

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

三、知识扩展

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

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

__stdcall (C++默认)

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

__cdecl (C语言默认)

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

参考资料

  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
此文转载自:http://www.cnblogs.com/chinazhangjie/archive/2012/08/18/2645475.html

你可能感兴趣的:(C language variable arguments)