变参函数——stdarg——printf——variable and function
va_list是一个宏,由va_start和va_end界定。
<wbr><wbr><wbr> typedef char* va_list;<br> void va_start ( va_list ap, prev_param );<br> type va_arg ( va_list ap, type );<br> void va_end ( va_list ap );<br><wbr><wbr><wbr> 其中,va_list 是一个字符指针,可以理解为指向当前参数的一个指针,取参必须通过这个指针进行。</wbr></wbr></wbr></wbr></wbr></wbr>
<wbr><wbr><wbr> <Step 1> 在调用参数表之前,应该定义一个 va_list 类型的变量,以供后用(假设这个 va_list 类型变量被定义为ap);<br><wbr><wbr><wbr> <Step 2> 然后对 ap 进行初始化,让它指向可变参数表里面的第一个参数。这是通过 va_start 来实现的,其第一个参数是 ap 本身,第二个参数是在变参表前面紧挨着的一个变量;<br><wbr><wbr><wbr> <Step 3> 然后是获取参数,调用 va_arg。它的第一个参数是 ap,第二个参数是要获取的参数的指定类型,并返回这个指定类型的值,同时把 ap 的位置指向变参表的下一个变量位置;<br><wbr><wbr><wbr> <Step 4> 获取所有的参数之后,我们有必要将这个 ap 指针关掉,以免发生危险,方法是调用 va_end。它是将输入的参数 ap 置为 NULL,应该养成获取完参数表之后关闭指针的习惯。</wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr>
<wbr></wbr>
例子:
int max(int n, ...)<wbr><wbr>// 定参 n 表示后面变参数量,定界用,输入时切勿搞错<br> {<wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><br><wbr>va_list ap;<wbr><wbr><wbr><wbr><wbr><wbr><wbr>// 定义一个 va_list 指针来访问参数表<br><wbr>va_start(ap, n);<wbr><wbr><wbr><wbr><wbr>// 初始化 ap,让它指向第一个变参<br><wbr>int maximum = -0x7FFFFFFF;<wbr><wbr><wbr>// 这是一个最小的整数<br><wbr>int temp;<br><wbr>for(int i = 0; i < n; i++)<br><wbr>{<br><wbr><wbr>temp = va_arg(ap, int);<wbr><wbr><wbr>// 获取一个 int 型参数,并且 ap 指向下一个参数<br><wbr><wbr>if (maximum < temp)<br><wbr><wbr><wbr>maximum = temp;<br><wbr>}<br><wbr>va_end(ap);<wbr><wbr><wbr><wbr><wbr><wbr><wbr>// 善后工作,关闭 ap<br><wbr>return maximum;<br> }</wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr>
// 在主函数中测试 max 函数的行为(C++ 格式)
int main()
{
<wbr>cout << max(3, 10, 20, 30) << endl;<br><wbr>cout << max(6, 20, 40, 10, 50, 30, 40) << endl;<br> }</wbr></wbr>
-------------------------------------------------------------------------------------------------------------------------
#include <stdio.h>
#include <stdarg.h>
void myputs(const char *str)
{
while(*str)
putchar(*str++);
}
void itoa(int nu, char *arr)
{
int i;
if(nu < 0)
{
arr[0] = '-';
arr[1] = '\0';
nu *= -1;
}
if(nu / 10)
itoa(nu / 10, arr);
for(i = 0; arr[i] != '\0'; i++);
arr[i] = nu % 10 + '0';
arr[i + 1] = '\0';
}
void itox(int nu, char *arr)
{
int i;
if(nu < 0)
{
arr[0] = '-';
arr[1] = '\0';
nu *= -1;
}
if(nu / 16)
itox(nu / 16, arr);
for(i = 0; arr[i]; i++);
if(nu % 16 > 9)
{
arr[i] = nu % 16 - 10 + 'a';
}
else
{
arr[i] = nu % 16 + '0';
}
arr[i + 1] = '\0';
}
int myprintf(const char *format, ...)
{
int data;
char *p;
char str[33] = {0};
va_list ap;
va_start(ap, format);
while(*format)
{
if(*format == '%')
{
format++;
switch(*format)
{
case 'c':
data = va_arg(ap, int);
putchar(data);
break;
case 'd':
data = va_arg(ap, int);
str[0] = '\0';
itoa(data, str);
myputs(str);
break;
case 'x':
data = va_arg(ap, int);
str[0] = '\0';
itox(data, str);
myputs(str);
break;
case 's':
p = va_arg(ap, char *);
myputs(p);
break;
case '%':
putchar('%');
break;
default:
break;
}
}
else
{
putchar(*format);
}
format++;
}
va_end(ap);
}
int main()
{
myprintf("%c hello %d 0x%xworld %s\n",
'A',
-1236,
0x1f,
"jack");
return 0;
}