关闭

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

标签: listunixmacrosoutputc
1153人阅读 评论(0) 收藏 举报
分类:

/* 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()有可能自己分配了内存需要释放。  
  }  
 

0
0

猜你在找
【套餐】Hadoop生态系统零基础入门
【套餐】嵌入式Linux C编程基础
【套餐】2017软考系统集成项目——任铄
【套餐】Android 5.x顶级视频课程——李宁
【套餐】深度学习入门视频课程——唐宇迪
【直播】广义线性模型及其应用——李科
【直播】从0到1 区块链的概念到实践
【直播】计算机视觉原理及实战——屈教授
【直播】机器学习之凸优化——马博士
【直播】机器学习&数据挖掘7周实训--韦玮
查看评论
* 以上用户言论只代表其个人观点,不代表CSDN网站的观点或立场
    个人资料
    • 访问:115716次
    • 积分:1389
    • 等级:
    • 排名:千里之外
    • 原创:10篇
    • 转载:50篇
    • 译文:0篇
    • 评论:14条
    最新评论