变参函数——stdarg——printf——variable and function

va_list是一个宏,由va_start和va_end界定。

    typedef char* va_list;
  void va_start ( va_list ap, prev_param );
  type va_arg ( va_list ap, type );
  void va_end ( va_list ap );
    其中,va_list 是一个字符指针,可以理解为指向当前参数的一个指针,取参必须通过这个指针进行。


    <Step 1> 在调用参数表之前,应该定义一个 va_list 类型的变量,以供后用(假设这个 va_list 类型变量被定义为ap);
    <Step 2> 然后对 ap 进行初始化,让它指向可变参数表里面的第一个参数。这是通过 va_start 来实现的,其第一个参数是 ap 本身,第二个参数是在变参表前面紧挨着的一个变量;
    <Step 3> 然后是获取参数,调用 va_arg。它的第一个参数是 ap,第二个参数是要获取的参数的指定类型,并返回这个指定类型的值,同时把 ap 的位置指向变参表的下一个变量位置;
    <Step 4> 获取所有的参数之后,我们有必要将这个 ap 指针关掉,以免发生危险,方法是调用 va_end。它是将输入的参数 ap 置为 NULL,应该养成获取完参数表之后关闭指针的习惯。

 

例子:

int max(int n, ...)  // 定参 n 表示后面变参数量,定界用,输入时切勿搞错
{                  
 va_list ap;       // 定义一个 va_list 指针来访问参数表
 va_start(ap, n);     // 初始化 ap,让它指向第一个变参
 int maximum = -0x7FFFFFFF;   // 这是一个最小的整数
 int temp;
 for(int i = 0; i < n; i++)
 {
  temp = va_arg(ap, int);   // 获取一个 int 型参数,并且 ap 指向下一个参数
  if (maximum < temp)
   maximum = temp;
 }
 va_end(ap);       // 善后工作,关闭 ap
 return maximum;
}

// 在主函数中测试 max 函数的行为(C++ 格式)
int main()
{
 cout << max(3, 10, 20, 30) << endl;
 cout << max(6, 20, 40, 10, 50, 30, 40) << endl;
}

-------------------------------------------------------------------------------------------------------------------------

#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;
}

 

你可能感兴趣的:(linux,printf,struct,C语言,starg)