变长参数列表函数
可變參數列表
頭文件<stdarg.h>提供了遍歷未知數目和類型的函數參數表的功能。該頭文件的實現因不同的機器而不同,但提供的接口是一致的。
假定函數 f 帶有可變數目的實際參數,lastarg 是它的最后一個命名的形式參數(參數列表必須至少包括一個命名參數)。那么,在函數 f 內聲明一個類型為 va_list 的變量 ap (argument pointer),它將依次指向每個實際參數。
?
va_list ap;
在訪問任何未命名的參數前,必須用 va_start 宏初始化 ap 一次。(va_start 將最后一個命名參數作為起點,將 ap 初始化為指向第一個未命名參數的指針)
此后,每次執行宏 va_arg 都將返回一個參數,并將 ap 指向下一個參數。 va_arg 使用一個類型名來決定返回的對象類型、指針移動的步長。
Type?va_arg(va_list ap,?Type);在所有的參數處理完畢之后,且在退出函數 f 之前,必須調用宏 va_end 以完成一些必要的清理工作。
void?va_end(va_list ap);Example
下面以實現函數printf的一個最簡單版本為例,介紹如何以可移植的方式編寫可處理變長參數列表的函數。因為我們的重點在于參數的處理,所以,函數minprintf只處理格式字符串和參數,格式轉換則通過調用函數printf實現。
函數printf的正確聲明形式為:
?
int?printf(char?*fmt,?...)其中,省略號表示參數表中參數的數量和類型是可變的。省略號只能出現在參數表的尾部。
因為minprintf函數不需要像printf函數一樣返回實際輸出的字符數,因此,我們將它聲明為下列形式:
編寫函數minprintf的關鍵在于如何處理一個甚至連名字都沒有的參數表。
#include <stdio.h>#include <stdarg.h>
/* minprintf 函數:帶有可變參數表的簡化的printf函數 */
void?minprintf(char?*fmt,?...)
{
? ? va_list ap;??/* 依次指向每個無名參數 */
? ??char?*p,?*sval;
? ??int?ival;
? ??double?dval;
? ??va_start(ap,?fmt);??/* 將 ap 指向第一個無名參數 */
? ??for?(p?=?fmt;?*p;?p++)?{
? ? ? ??if?(*p?!=?'%')?{
? ? ? ? ? ??putchar(*p);
? ? ? ? ? ??continue;
? ? ? ??}
? ? ? ??switch?(*++p)?{
? ? ? ? ? ??case?'d':
? ? ? ? ? ? ? ? ival?=?va_arg(ap,?int);
? ? ? ? ? ? ? ??printf("%d",?ival);
? ? ? ? ? ? ? ??break;
? ? ? ? ? ??case?'f':
? ? ? ? ? ? ? ? dval?=?va_arg(ap,?double);
? ? ? ? ? ? ? ??printf("%f",?dval);
? ? ? ? ? ? ? ??break;
? ? ? ? ? ??case?'s':
? ? ? ? ? ? ? ??for?(sval?=?va_arg(ap,?char?*);?*sval;?sval++) putchar(*sval);
break;
default:
putchar(*p);
break;
}
}
va_end(ap);/* 結束時的清理工作 */
}
轉載于:https://www.cnblogs.com/qq78292959/archive/2012/06/05/2536459.html
總結