VA_LIST的用法

有关VA_LIST的用法--变参函数的实现


VA_LIST 是在C语言中解决变参问题的一组宏 VA_LIST的用法:      
       (1)首先在函数里定义一具VA_LIST型的变量,这个变量是指向参数的指针
       (2)然后用VA_START宏初始化变量刚定义的VA_LIST变量,这个宏的第二个参数是第一个可变参数的前一个参数,是一个固定的参数。
       (3)然后用VA_ARG返回可变的参数,VA_ARG的第二个参数是你要返回的参数的类型。
       (4)最后用VA_END宏结束可变参数的获取。然后你就可以在函数里使用第二个参数了。如果函数有多个可变参数的,依次调用VA_ARG获取各个参数。 VA_LIST在编译器中的处理: (1)在运行VA_START(ap,v)以后,ap指向第一个可变参数在堆栈的地址。
(2)VA_ARG()取得类型t的可变参数值,在这步操作中首先apt = sizeof(t类型),让ap指向下一个参数的地址。然后返回ap-sizeof(t类型)的t类型*指针,这正是第一个可变参数在堆栈里的地址。然后用*取得这个地址的内容。
(3)VA_END(),X86平台定义为ap = ((char*)0),使ap不再指向堆栈,而是跟NULL一样,有些直接定义为((void*)0),这样编译器不会为VA_END产生代码,例如gcc在Linux的X86平台就是这样定义的。 要注意的是:由于参数的地址用于VA_START宏,所以参数不能声明为寄存器变量,或作为函数或数组类型。 使用VA_LIST应该注意的问题:
   (1)因为va_start, va_arg, va_end等定义成宏,所以它显得很愚蠢,可变参数的类型和个数完全在该函数中由程序代码控制,它并不能智能地识别不同参数的个数和类型. 也就是说,你想实现智能识别可变参数的话是要通过在自己的程序里作判断来实现的.
   (2)另外有一个问题,因为编译器对可变参数的函数的原型检查不够严格,对编程查错不利.不利于我们写出高质量的代码。
小结:可变参数的函数原理其实很简单,而VA系列是以宏定义来定义的,实现跟堆栈相关。我们写一个可变参数的C函数时,有利也有弊,所 以在不必要的场合,我们无需用到可变参数,如果在C++里,我们应该利用C++多态性来实现可变参数的功能,尽量避免用C语言的方式来实现。 示例代码:

#include<stdio.h>
#include<stdarg.h>
void simple_va_fun(int start,...)
{
     va_list arg_ptr;
     int nArgValue = start;
     int nArgCount = 0;
     va_start(arg_ptr,start);
     do
     {
     ++nArgCount;
      printf("the %d the arg:%d\n",nArgCount,nArgValue);
       nArgValue=va_arg(arg_ptr,int);
    }while(nArgValue != -1);
    return; 
     }
main()
{
simple_va_fun(100,-1);
simple_va_fun(100,200,-1);
getchar();
      }

///

int writeMultiString(void *c, ...)
{
    va_list ap;
    const char *str;

    va_start(ap, c);
    str = va_arg(ap, const char *);

    while (str != NULL)
    {
        fputs(str, (FILE *)c);
        str = va_arg(ap, const char *);
    }

    va_end(ap);
    return 0;
}
引用:  va_end() 对由 va_start() 或者 va_copy() 初始化的 va_list 类型对象的执行清理过程。va_end() 会修改 va_list 对象实例,使其不再可用。 如果没有相对应的对 va_start() 或者 va_copy() 函数的调用,或者如果在一个函数调用 va_start() 或者 va_copy() 函数之前没有调用 va_end() ,行为没有定义,即在这些情况下,没有定义如何去处理。 引用: va_list 是一个类型,用于表示可数列表。va_start() 是一个函数,用于初始化 va_list 类型的对象。va_copy() 也是一个函数,用于将已有的 va_list 类型对象内容拷贝到另一个 va_list 类型对象。va_arg() 是一个函数,用于获取 va_list 对象指针指向位置的下一个数的值。 根据引用和引用的内容,va_list 是用于表示可数列表的类型,而 va_start() 是用于初始化 va_list 对象的函数。通过调用 va_start() 函数,我们可以将一个 va_list 对象和可数列表的第一个数进行关联。这个对象可以被之后的操作所使用。 <span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* *3* [C/C++可数列表数处理方法va_listva_start()、va_copy()、va_arg()、va_end()](https://blog.youkuaiyun.com/StoryZX/article/details/125748675)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 100%"] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值