1. 程式人生 > >對巨集定義:_INTSIZEOF(n)、va_start(ap,v)、va_arg(ap,t)、va_end(ap)的理解

對巨集定義:_INTSIZEOF(n)、va_start(ap,v)、va_arg(ap,t)、va_end(ap)的理解

程式碼:

typedef char *  va_list;
#define _INTSIZEOF(n)   ( (sizeof(n) + sizeof(int) - 1) & ~(sizeof(int) - 1) )
#define va_start(ap,v)  ( ap = (va_list)&v + _INTSIZEOF(v) )
#define va_arg(ap,t)    ( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) )
#define va_end(ap)      ( ap = (va_list)0 )

理解:

1:
#define _INTSIZEOF(n)   ( (sizeof(n) + sizeof(int) - 1) & ~(sizeof(int) - 1) )
目的是把sizeof(n)的結果變成至少是sizeof(int)的整倍數,一般用來在結構中實現按int的倍數對齊。

2:
#define va_start(ap,v)  ( ap = (va_list)&v + _INTSIZEOF(v) )
將ap指標指向引數v的下一引數地址處。

3:
 #define va_arg(ap,t)    ( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) )
這條程式碼目的是將指標移動到下一引數後,再返回上一地址將上一地址的值取出。原理:根據運算順序會先執行(ap += _INTSIZEOF(t))相當於ap=ap+_INTSIZEOF(t),也就是將指標ap指向下一地址;然後再執行- _INTSIZEOF(t)回到原地址,並轉化為型別*的指標,再取指標所指地址的值。

4:
#define va_end(ap)      ( ap = (va_list)0 )
結束指標ap的使用,避免出現野指標。