zoukankan      html  css  js  c++  java
  • C语言利用va_list、va_start、va_end、va_arg宏定义可变參数的函数

    在定义可变參数的函数之前,先来理解一下函数參数的传递原理:

    1、函数參数是以栈这样的数据结构来存取的,在函数參数列表中,从右至左依次入栈。

    2、參数的内存存放格式:參数的内存地址存放在内存的堆栈段中,在运行函数的时候,从最后一个(最右边)參数開始入栈。因此栈底高地址,栈顶低地址,举个样例说明一下:

    void test(int a, float b, char c);

    那么,在调用test函数的时候,实參char c先进栈,然后是float b,最后才是int a,因此在内存中变量的存放次序是c->b->a,由于从理念上来说,我们仅仅要探測到随意一个变量的地址,而且知道其他变量的类型,通过指针移位运算,就能够顺藤摸瓜找到其他的输入变量。

    实现一个可变參数的函数,须要用到下面几个宏:

    typedef char* va_list;	// 用于声明一个指向參数列表的字符型指针变量
    void va_start(va_list ap, prev_param);	// 第一个參数为指向可变參数字符指针变量,第二个參数是可变參数的第一个參数,通经常使用于指定可变參数列表中參数的个数
    void va_arg(va_list ap, type);	// 第一个參数为指向可变參数字符指针变量,第二个參数是可变參数的数据类型
    void va_end(va_list ap);// 将存放可变參数字符串的变量清空(赋值为NULL)
    3、演示样例:求N个数的和

    int sum(int count, ...)
    {
        int sum = 0;
        int i;
        va_list ap;
        va_start(ap, count);
        for (i = 0; i < count; ++i)
        {
            sum += va_arg(ap, int);
        }
        va_end(ap);
        return sum;
    }
    int main(int argc, const char * argv[])
    {
        int ret = sum(5, 1, 2, 3, 4, 5);
        printf("sum: %d
    ",ret);
    }
    4、演示样例:标准库可变參数的相关函数使用

    void test(int count,...)
    {
        va_list ap;
        va_start(ap, count);
        vprintf("%d,%d,%d
    ", ap);                  // 格式化输出可变參数的值
        
        char buff[1024];
        vsprintf(buff, "a=%d,b=%d,c=%d
    ", ap);     // 将可变參数列表的值格式化输出到缓冲区中
        printf("%s
    ",buff);
        
        vfprintf(stdout, "a=%d,b=%d,c=%d
    ", ap);   // 将可变參数列表的值打印到标准输出中
        
        // 可变參数传入的是int类型的地址
        vsscanf("10,30,40", "%d,%d,%d", ap);        // 依次将格式化字符串的值,输入到可变參数变量中
        
        vsnprintf(buff, 30, "a=%d,b=%d,c=%d", ap);  // 将可变參数列表的值,格式化输出指定长度(30个字符)的字符串到缓冲区中
        printf("vsnprintf=%s
    ",buff);
        
        va_end(ap);
    }

  • 相关阅读:
    内置函数二
    内置函数一
    lambda表达式
    函数参数
    set集合
    元组和字典的功能
    列表功能介绍
    分篮子
    松鼠配对?
    奇数次的数?
  • 原文地址:https://www.cnblogs.com/mengfanrong/p/4306034.html
Copyright © 2011-2022 走看看