C可变参数详解及其函数设计
在stdarg.h文件中有如下幾個宏定義:
[cpp]view plaincopy#include <vadefs.h>
#define va_start _crt_va_start
#define va_arg _crt_va_arg
#define va_end _crt_va_end
#endif ?/* _INC_STDARG */
其定義在vadefs.h中分別為:
#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 ) // 將指針置為無效
在進程中,堆棧地址是從高到低分配的.當執行一個函數的時候,將參數列表入棧,壓入堆棧的高地址部分,然后入棧函數的返回地址,接著入棧函數的執行代碼,這個入棧過程,堆棧地址不斷遞減,一些***就是在堆棧中修改函數返回地址,執行自己的代碼來達到執行自己插入的代碼段的目的.
總之,函數在堆棧中的分布情況是:地址從高到低,依次是:函數參數列表,函數返回地址,函數執行代碼段.
堆棧中,各個函數的分布情況是倒序的.即最后一個參數在列表中地址最高部分,第一個參數在列表地址的最低部分.參數在堆棧中的分布情況如下:
? ? ? 最后一個參數
倒數第二個參數
...
第一個參數
函數返回地址
函數代碼段
代碼示例:這里是一個可變參數的加法函數
#include<stdio.h>
#include<stdlib.h>
#include<stdarg.h>
/*功能:可變參數求和
*參數:numCount參數個數 ...可變求和參數
*返回值:參數相加的和
*/
int sum(int numCount,...) ?
{ ?
int result = 0; ? ? //計算結果
va_list ap; ? ? ? ? //初始化指向可變參數列表的指針(typedef char* va_list)
? ?va_start(ap,numCount); ? ? ? ? ?//將第一個可變參數的地址付給ap,即ap指向可變參數列表的開始
for(int i = 0;i<numCount;i++) ?
? ? ? ?result += va_arg(ap,int); ? //得到第一個可變參數的值,并且ap指針上移一個_INTSIZEOF(int),即指向下一個可變參數的地址.
? ?va_end(ap); ? ? ? ? ? ? ? ? ? ? //置空ap,即ap=(void *)0;
return result; ?
} ?
int main(void) ?
{ ?
? ?printf("20+15+3+8=%d\n",sum(4,20,15,3,8)); ?
? ?system("pause"); ?
return 0; ?
} ?
轉載于:https://blog.51cto.com/7212823/1202392
總結
以上是生活随笔為你收集整理的C可变参数详解及其函数设计的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: IE滤镜实现透明度/阴影/渐变等特效
- 下一篇: hdu 1044