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

原创 2007年09月24日 10:43:00

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

C/C++中va_list,va_start,va_arg,va_end的使用和原理

参考: http://www.cnblogs.com/dongsheng/p/4001555.html http://www.cnblogs.com/yongssu/p/4677556.html...
  • piaoxuezhong
  • piaoxuezhong
  • 2017年03月06日 13:05
  • 936

va_list、va_start、va_arg、va_end宏的使用(转)

当你的函数的参数个数不确定时,就可以使用上述宏进行动态处理,这无疑为你的程序增加了灵活性。Example:CString AppendString(CString str1,...)//一个连接字符串...
  • tuwen
  • tuwen
  • 2008年03月21日 14:18
  • 2352

va_list是一个宏,由va_start和va_end界定,一时难说清,详细见《Windows32程序设计》Unicode部分

va_list是一个宏,由va_start和va_end界定,一时难说清,详细见《Windows32程序设计》Unicode部分va_list   structure       Used   to ...
  • wenzuodan
  • wenzuodan
  • 2008年01月31日 15:50
  • 4065

va_list、va_start、va_arg、va_end等利用宏定义实现可变参数

#include //#include //#include //int sum(int n, ...); typedef void* va_list; #define _INTSIZEOF...
  • z251941074
  • z251941074
  • 2015年01月31日 21:13
  • 1386

C/C++(6)va_list、va_start、va_arg、va_end实现可变参数列表函数

经典的可变参数表printf函数的实现原理:在C/C++中,对函数参数的扫描是从后向前的。 函数参数是通过压入堆栈的方式来给函数传参数的,最先压入的参数最后出来,而栈是从内存的高地址向低地址存放的,...
  • qjpxAAA
  • qjpxAAA
  • 2016年10月24日 17:40
  • 409

C语言可变参数函数详解 va_list va_arg va_start宏

可变参数函数: 参数数目是可变的。比如 C 语言的 printf 系列的(格式化输入输出等)函数,都是参数可变的,形式是形参是三个点。 如printf 函数的声明: int printf ( c...
  • geqiandebei
  • geqiandebei
  • 2015年01月05日 15:36
  • 1431

C/C++中用va_start/va_arg/va_end实现可变参数函数的原理与实例详解 .

原文链接:http://blog.csdn.net/slvher/article/details/9881171   在C/C++中,我们经常会用到可变参数的函数(比如printf/snprintf等...
  • LoveJiaYu
  • LoveJiaYu
  • 2016年11月23日 19:48
  • 1064

ios 可变参数(va_list,va_start,va_end)

原文地址:ios 可变参数(va_list,va_start,va_end) 例如:UIAlertView的init方法中的otherButtonTitles:(NSString *)oth...
  • u012894479
  • u012894479
  • 2015年06月23日 15:18
  • 1732

iOS开发可变参数va_list、va_start、va_end

iOS 可变参数(va_list,va_start,va_end) 例如:UIAlertView的init方法中的otherButtonTitles:(NSString *)otherButtonT...
  • wgl_happy
  • wgl_happy
  • 2015年12月11日 11:22
  • 455

va_list,va_start,va_arg,va_end可变参数

  • paomadi
  • paomadi
  • 2013年01月24日 21:01
  • 1929
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:va_start,va_arg,va_end,va_list应用举例--实现可变参数的函数
举报原因:
原因补充:

(最多只允许输入30个字)