va_start,va_arg,va_end,va_list应用举例--实现可变参数的函数

/* VA.C: The program below illustrates passing a variable
 * number of arguments using the following macros:
 *      va_start            va_arg              va_end
 *      va_list             va_dcl (UNIX only)
 */

#include <stdio.h>
#define ANSI            /* Comment out for UNIX version     */
#ifdef ANSI             /* ANSI compatible version          */
#include <stdarg.h>
int average( int first, ... );
#else                   /* UNIX compatible version          */
#include <varargs.h>
int average( va_list );
#endif

void main( void )
{
   /* Call with 3 integers (-1 is used as terminator). */
   printf( "Average is: %d/n", average( 2, 3, 4, -1 ) );

   /* Call with 4 integers. */
   printf( "Average is: %d/n", average( 5, 7, 9, 11, -1 ) );

   /* Call with just -1 terminator. */
   printf( "Average is: %d/n", average( -1 ) );
}

/* Returns the average of a variable list of integers. */
#ifdef ANSI             /* ANSI compatible version    */
int average( int first, ... )
{
   int count = 0, sum = 0, i = first;
   va_list marker;

   va_start( marker, first );     /* Initialize variable arguments. */
   while( i != -1 )
   {
      sum += i;
      count++;
      i = va_arg( marker, int);
   }
   va_end( marker );              /* Reset variable arguments.      */
   return( sum ? (sum / count) : 0 );
}
#else       /* UNIX compatible version must use old-style definition.  */
int average( va_alist )
va_dcl
{
   int i, count, sum;
   va_list marker;

   va_start( marker );            /* Initialize variable arguments. */
   for( sum = count = 0; (i = va_arg( marker, int)) != -1; count++ )
      sum += i;
   va_end( marker );              /* Reset variable arguments.      */
   return( sum ? (sum / count) : 0 );
}
#endif

Output
Average is: 3
Average is: 8
Average is: 0

 printf函数的实现比较的复杂,但是大体上是这样的。  
   
  函数用到两个比较特别的类型称之为   va_list   它主要用来访问调用函数时的参数,<其实说穿了也就是通过访问栈来实现的>  
  由于printf函数的参数是可变数目的,所以首先通过va_list变量来访问第一的参数也就是一个可以肯定的参数-字符串指针。  
   
  大概的实现我随便写写,不一定对,但是思路就是这样了。  
  int     printf(   const     char       *szFormat,...   //声明可变数目的参数   )  
  {  
          va_list         vaOldStack;  
          va_start(   vaOldStack   );   //初始化变量  
                                                             
          const   char       *szAnalysis   =   va_arg(   vaOldStack,   char   *   );  
   //存取了第一个参数,vaOldStack被更新指向下一个参数的位置  
           
          //   开始分析字符串    
          while(   szAnalysis[   i   ]   !=   '/0'   )  
          {  
                  if(   szAnalysis[   i   ]   ==   '%'   )   //   判定格式  
                  switch(   szAnalysis[   i   +   1   ]   )  
                  {  
                        case   'c':   //是要求输出字符  
                            int   ch   =   va_arg(   vaOldStack,   char   )   //从栈中取两个字节,  
               //vaOldStack被更新,指向下一个参数地址。  
                            putch(   ch   );  
                            ++i;  
                            break;  
   
                      case     'd':   //是要求输出整数  
                            int   Value   =   va_arg(   vaOldStack,   int   )//   同样取两个字节  
                            //   处理整数输出...  
                            ++i;  
                            break;  
   
                      case     'f':   //    
                      ......................  
                            以上只考虑到一小部分情况,实际还有可能要处理格式字符如  
  "%6d"   这样的。这些函数都要自己编写处理,有些比较复杂,有些简单   如处理字符输出。  
                    default:  
                        putch(szAnalysis[   i   ]   );    
                  }  
            ++i;    
        }  
          va_end(   va_list   );   //加上这句才行,因为va_start()有可能自己分配了内存需要释放。  
  }  
 

阅读更多
个人分类: 基础知识
想对作者说点什么? 我来说一句

没有更多推荐了,返回首页

加入CSDN,享受更精准的内容推荐,与500万程序员共同成长!
关闭
关闭