C语言可变参数函数的两种实现方式以及va_list、va_start、va_arg、va_end详解

函数的可变参数的实现方式有两种:

1.通过库函数stdarg.h实现

先贴代码

int add(int n,...)
{
    int i = 0;
    int sum = 0;
    va_list ap;
    va_start(ap,n);
    for(i = 0;i < n;i++)
    {
        sum += va_arg(ap,int);
    }
    va_end(ap);
    return sum;

}

在函数中

va_list ap;//首先声明一个char类型的变量指针

va_start(ap,n);//将指针指向第二个参数,没错就是函数的第二个参数,va_start的宏定义是通过第一个参数定位到第二个参数的地址

va_arg(ap,int);//获取当前函数的值,并且指针+1(指针指向下一个参数)

va_end(ap);结束

2.不需通过库函数,直接通过地址来实现

先贴代码

int add1(int n,...)
{
    int *arg = (int *)&n + 1;
    int i = 0;
    int sum = 0;
    for(i = 0;i < n;i++)
    {
        sum += *arg;
        arg++;
    }
    return sum;
}
由于函数的参数是通过堆栈存储的,而且是把函数参数从右向左一次压入堆栈,所以说,第一个参数在堆顶,我们就可以通过地址依次获取参数

具体参考文章:https://blog.csdn.net/Bugggget/article/details/79519395

最后贴出完整的代码:

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

int add(int n,...);
int add1(int n,...);

int main()
{
    printf("Hello world!\n");
    printf("通过库函数实现的add(3,10,20,30) = %d\n",add(3,10,20,30));
    printf("通过堆栈地址实现的add1(4,50,60,20,30) = %d\n",add1(4,50,60,20,30));
    return 0;
}
int add(int n,...)
{
    int i = 0;
    int sum = 0;
    va_list ap;
    va_start(ap,n);
    for(i = 0;i < n;i++)
    {
        sum += va_arg(ap,int);
    }
    va_end(ap);
    return sum;

}
int add1(int n,...)
{
    int *arg = (int *)&n + 1;
    int i = 0;
    int sum = 0;
    for(i = 0;i < n;i++)
    {
        sum += *arg;
        arg++;
    }
    return sum;
}

C语言可变参数函数的两种实现方式以及va_list、va_start、va_arg、va_end详解