《C标准库》——之<stdarg.h>

C语言有个很强大的功能,依靠它,实现了printf等这类有着变长参数列表的函数或者宏。它就是在<stdarg.h>里的变长参数。


内容:

va_list   :它是一个适合保存va_start、va_arg和va_end所需要的信息的类型。
va_start  :void va_start(va_list ap, parmN); 要在访问所有未命名的参数之前调用宏va_start,宏va_start对ap进行初始化,以便va_arg和va_end使用。
va_arg    :type va_arg(va_list, type); 它展开是一个表达式,这个表达式的类型和值跟调用的下一个参数的相同。参数ap应该和va_start初始化的va_list ap相同。
va_end    :如果没有执行va_start,或者在返回之前没有调用va_end,那么这种行为未定义。


实现:


还要用到我们在<stddef.h>中提到的<yvals.h>,它里面有两个宏:_AUPBND和_ADNBND

   #define _AUPBND (sizeof (acpi_native_int) - 1) //acpi_native_int 为4字节(32位)(根据机子字数而定)
   #define _ADNBND (sizeof (acpi_native_int) - 1) 

typedef char* va_list; // 因此,va_list实际上表示的是char*
#define va_start(ap, A)  (void)((ap) = (char*)&(A) + _Bnd(A, _AUPBND))
#define va_arg(ap, T)    (*(T*) (((ap) += _Bnd(T, _AUPBND)) - _Bnd(T, _ADNBND)))
#define va_end(ap)       (void)0

#define _Bnd(X, bnd)    (sizeof (X) + (bnd) &~(bnd) )




在实现里我们看到,va_list其实是char*类型,va_start将ap移动到参数列表的首地址,为va_arg的实现做准备,va_end啥都没干。

而va_arg是技巧性最强的一个。首先通过增加va_list对象的内容来使它跳向下一个参数空间的起始位置,然后然退回来指向当前参数的起始位置,然后通过强制类型住哪换把这个指针转换为指定类型的指针,最后解引用获得这个指针以访问存储在数据对象中的值。


在va_start和va_arg里,都用到了一个_Bnd(T, _AUPBND)或_Bnd(T, _ADNBND),由_AUPBND和_ADNBND的宏定义我们知道,它们总是等于在指定的机器中int类型的字节数减1。

事实上, _Bnd宏做的事情就是字节对齐,C语言函数入栈都是字节对齐的,都是所在机器上int类型字节的整数倍进行压栈,这样做可以是内存高效的访问。

sizeof(X)先计算出X类型的大小(假设为s),(bnd) & ~(bnd) 其实bnd就是_AUPBND和_ADNBND(我们假设它们都为3),先将X的大小加3,加了3就会清掉s的二进制数的最低两位,做个假设,假设s为7:00000111,然后加3:00000011,就变为00001010,最后&3的反:(11111100),最后_Bnd计算的结果就是00001000,为8,是int类型大小的整数倍,实现了字节对齐。









  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是<Arduino.h>头文件的代码示例: ```c++ #ifndef Arduino_h #define Arduino_h #include <stdlib.h> #include <stdbool.h> #include <string.h> #include <math.h> #include <stdint.h> #include <stdio.h> #include <stdarg.h> #include "binary.h" #include "itoa.h" #include "pins_arduino.h" #include "wiring.h" #include "wiring_analog.h" #include "wiring_digital.h" #include "wiring_pulse.h" #include "wiring_shift.h" #include "WInterrupts.h" #ifdef __cplusplus extern "C"{ #endif void setup(void); void loop(void); #ifdef __cplusplus } // extern "C" #endif #endif ``` 以下是<inttypes.h>头文件的代码示例: ```c++ #ifndef _INTTYPES_H_ #define _INTTYPES_H_ #include <stdint.h> #define PRId8 "d" #define PRId16 "d" #define PRId32 "ld" #define PRId64 "lld" #define PRIi8 "i" #define PRIi16 "i" #define PRIi32 "li" #define PRIi64 "lli" #define PRIo8 "o" #define PRIo16 "o" #define PRIo32 "lo" #define PRIo64 "llo" #define PRIu8 "u" #define PRIu16 "u" #define PRIu32 "lu" #define PRIu64 "llu" #define PRIx8 "x" #define PRIx16 "x" #define PRIx32 "lx" #define PRIx64 "llx" #define PRIX8 "X" #define PRIX16 "X" #define PRIX32 "lX" #define PRIX64 "llX" #define PRIdPTR "ld" #define PRIiPTR "li" #define PRIoPTR "lo" #define PRIuPTR "lu" #define PRIxPTR "lx" #define PRIXPTR "lX" #define SCNd8 "hhd" #define SCNd16 "hd" #define SCNd32 "ld" #define SCNd64 "lld" #define SCNi8 "hhi" #define SCNi16 "hi" #define SCNi32 "li" #define SCNi64 "lli" #define SCNo8 "hho" #define SCNo16 "ho" #define SCNo32 "lo" #define SCNo64 "llo" #define SCNu8 "hhu" #define SCNu16 "hu" #define SCNu32 "lu" #define SCNu64 "llu" #define SCNx8 "hhx" #define SCNx16 "hx" #define SCNx32 "lx" #define SCNx64 "llx" #define SCNdPTR "ld" #define SCNiPTR "li" #define SCNoPTR "lo" #define SCNuPTR "lu" #define SCNxPTR "lx" #endif ``` 以下是<Stream.h>头文件的代码示例: ```c++ /* Stream.h - library for stream base class Copyright (C) 2006-2009 David A. Mellis Modified 23 November 2010 by Mark Sproul Modified 3 December 2017 by Chuck Todd */ #ifndef Stream_h #define Stream_h #include "Print.h" #define DEC 10 #define HEX 16 #define OCT 8 #define BIN 2 class Stream : public Print { public: virtual int available() = 0; virtual int read() = 0; virtual int peek() = 0; virtual void flush() = 0; }; #endif ```

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值