可变参数宏va_arg(ap, T)解释记录

/*32bit system*/
typedef s32 acpi_native_int;

/* Storage alignment properties */
#define  _AUPBND                (sizeof (acpi_native_int) - 1)
#define  _ADNBND                (sizeof (acpi_native_int) - 1)

/* Variable argument list macro definitions */
#define _bnd(X, bnd)            (((sizeof (X)) + (bnd)) & (~(bnd)))
#define va_arg(ap, T)           (*(T *)(((ap) += (_bnd (T, _AUPBND))) - (_bnd (T,_ADNBND))))
#define va_end(ap)              (ap = (va_list) NULL)
#define va_start(ap, A)         (void) ((ap) = (((char *) &(A)) + (_bnd (A,_AUPBND))))
  • 该宏位于stdarg.h中,定义了可变参数的地址;
  • 可变参数是充分利用在函数传参时,会将参数依次压入堆栈,然后根据堆栈的特点,访问可变参数;
  • 以32位系统为例,_AUPBND和_ADNBND表示系统的对齐方式,在32位系统中为3
  • _bnd(X, bnd)是要求出X在堆栈中占的内存大小,注意是内存对齐以后的大小;
  • va_arg(ap, T)有两个作用,第一:返回当前可变参数的地址;第二:将ap指针向后移动到指向第二个可变参数;
  • va_end(ap)表示将ap赋空,防止出现野指针;
  • va_start(ap, A)表示将ap指向第一个可变参数;
  • 在看这几个宏的时候,尤其有点不理解va_arg,觉得将ap先加一个值,再减去一个相等的值,不是没变化吗?可是在实际写了个小函数测试时才发现“+=”符号的妙用,第一次赋值是移动指针ap的位置,第二次减去相同的值时并未改变指针ap的指向,返回的值表示的时表达式的值,还是原来ap的指向。
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
va_start、va_argva_end是C语言中用来传递可变数量参数的。它们通常与可变参数函数一起使用,例如printf函数。 首先,我们来介绍va_start。它的作用是初始化一个指向可变参数列表的指针,以便访问参数列表中的参数。具体语法如下: ```c void va_start(va_list ap, last_param); ``` 其中,ap是一个指向可变参数列表的指针,last_param是最后一个固定参数。va_start会在last_param之后找到可变参数列表的起始位置,并将ap指向该位置。 接下来是va_arg,它用于获取可变参数列表中的参数值。具体语法如下: ```c type va_arg(va_list ap, type); ``` 其中,ap是一个指向可变参数列表的指针,type是要获取的参数的类型。va_arg会返回ap指向的参数值,并将ap移动到下一个参数的位置。 最后是va_end,它用于结束对可变参数列表的访问。具体语法如下: ```c void va_end(va_list ap); ``` 调用va_end后,ap指针将不再有效。 使用这三个,我们可以实现对可变数量参数的遍历和访问。例如,在printf函数中使用可变参数列表,可以通过va_start、va_argva_end来依次获取不同类型的参数,并按照格式字符串进行输出。 需要注意的是,可变参数的传递是通过栈来实现的,所以对于不同的硬件平台和编译器,可变参数的传递方式可能有所不同。因此,在使用这些时,需要参考相关平台和编译器的文档进行正确的使用。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值