像 printf 这种多参函数, 是借用 stdarg.h 中的宏实现的.
va_list  : 用于定义遍历参数列表的指针;
va_start : 让指针指向第一个参数;
va_arg   : 获取下一个参数, 并向后移动一个位置;
va_end   : 释放指针, 完成遍历.


1. 整数求和:

本例实现了对系列整数求和, 要求至少要有三个参数, 并且最后一个必须是 0.
最后的 0 用于识别列表结束.
#include <stdio.h>
#include <stdarg.h>

int sum(int n1, int n2, ...)
{
    /* 定义一个指向参数列表的指针, 必须是 va_list 类型 */
    va_list p;

    /* 定义输出变量, 并先获取前两个值 */
    int out = n1 + n2;

    /* 把指针指向最后一个明确的变量 */
    va_start(p, n2);

    /* 用 va_arg 获取下一个整数值, va_arg 会同时把指针向后移动整数大小的位置 */
    /* 本例是假定参数都是整数值, 遇 0 终止; 这样在使用是最后一个参数必须是 0 */
    while ((n2 = va_arg(p, int)) != 0) out += n2;
   
    /* 结束 */
    va_end(p);

    return(out);
}

int main(void)
{
    printf("%d\n", sum(2,2,2,0));             /* 6  */
    printf("%d\n", sum(1,2,3,4,5,6,7,8,9,0)); /* 45 */    
    getchar();
    return 0;
}


2. 整数求和(修改版):

这个函数要求至少要两个参数, 最后必须是 0.
#include <stdio.h>
#include <stdarg.h>

int sum(int n1, ...)
{
    va_list p;
    int out = n1;
    va_start(p, n1);
    while ((n1 = va_arg(p, int)) != 0) out += n1;
    va_end(p);
    return(out);
}

int main(void)
{
    printf("%d\n", sum(2,0));                 /* 2  */
    printf("%d\n", sum(1,2,3,4,5,6,7,8,9,0)); /* 45 */    
    getchar();
    return 0;
}


3. 指定数目的浮点数求和:
该函数参数一指定数目, 之后是列表.
#include <stdio.h>
#include <stdarg.h>

double sum(int num, double f1, ...)
{
    va_list p;
    double out = f1;
    va_start(p, f1);
    while(--num)
    {
        f1 = va_arg(p, double);
        out += f1;
    }     
    va_end(p);
    return(out);
}

int main(void)
{
    printf("%g\n", sum(3, 1.1, 2.2, 3.3));  /* 6.6 */
    printf("%g\n", sum(2, 1.1, 2.2, 3.3));  /* 3.3 */    
    getchar();
    return 0;
}


4. 整数与浮点数求和:

本例模拟了 printf 函数的样式, 但只支持整数与浮点数.
#include <stdio.h>
#include <stdarg.h>

double sum(char *str, ...)
{
    va_list List;
    
    double out = 0.0;

    va_start(List, str); 

    while (*str)           
    {
        if (*str == '%') 
        {
            switch (*(++str)) {
                case 'd': out += va_arg(List, int);    break;
                case 'f': out += va_arg(List, double); break;
            }
        }
        str++;
    }

    va_end(List);
    return(out);
}


int main(void)
{
    double d = sum("%d,%f,%d,%f", 1, 1.1, 2, 2.2);

    printf("%g\n", d); /* 6.3 */
    getchar();
    return 0;
}


posted on 2008-12-04 22:20  万一  阅读(2391)  评论(0编辑  收藏  举报