C语言中可变参数函数的使用

由于在C语言中没有函数重载,解决不定数目函数参数问题变得比较麻烦;即使采用C++,如果参数个数不能确定。也很难采用函数重载。对这种情况,有些人采用指针参数来解决问题。本文就C语言中对不定参数函数的使用方法进行小结。

 

1、函数参数在堆栈中的分布

地址从高到低,依次是:函数参数列表,函数返回地址,函数执行代码段。堆栈中,各个函数的分布情况是倒序的,即最后一个参数在列表中地址最高部分,第一个参数在列表地址的最低部分。

 

2VA_LIST宏是在C语言中解决变参问题的一组宏,在C语言的stdarg.h头文件中有一下几个定义:

1_INTSIZEOF 宏,获取类型占用的空间长度,最小占用长度为int的整数倍

 #define _INTSIZEOF(n)   ((sizeof(n)+sizeof(int)-1)&~(sizeof(int) - 1) )

2VA_START宏,获取可变参数列表的第一个参数的地址(ap是类型为va_list的指针,v是可变参数最左边的参数

#define va_start(ap,v) ( ap = (va_list)&v + _INTSIZEOF(v) ) 

3VA_ARG宏,获取可变参数的当前参数,返回指定类型并将指针指向下一参数(t参数描述了当前参数的类型)

#define va_arg(ap,t) ( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) )

4VA_END宏,清空va_list可变参数列表

#define va_end(ap)    ( ap = (va_list)0 ) 

 

3VA_LIST使用方法:

1)首先在函数里定义一具VA_LIST型的变量,这个变量是指向参数的指针;
2)然后用VA_START宏初始化变量刚定义的VA_LIST变量;
3)然后用VA_ARG返回可变的参数,VA_ARG的第二个参数是你要返回的参数的类型(如果函数有多个可变参数的,依次调用VA_ARG获取各个参数);
4)最后用VA_END宏结束可变参数的获取。

 

4、使用可变参数需要注意的问题:

1)可变参数的类型和个数完全由程序代码控制,它并不能智能地识别不同参数的个数和类型;
2)如果我们不需要一一详解每个参数,只需要将可变列表拷贝至某个缓冲,可用vsprintf函数;
3)因为编译器对可变参数的函数的原型检查不够严格,对编程查错不利。不利于我们写出高质量的代码

(4) 可变参数是由宏实现的,但是由于硬件平台的不同,编译器的不同,宏的定义也不相同。

 

5、例子:

1)打印字符串

 

#include "stdafx.h" #include <iostream> #include <string> #include <stdarg.h> using namespace std; void printstrings(int num,...) { va_list arg_ptr; va_start(arg_ptr,num); while (num--) { printf("/n%s",va_arg(arg_ptr,char*)); } va_end(arg_ptr); } int main(int argc, char* argv[]) { printstrings(7,"hello","this","is","uncertain","number","parameter","example"); return 0; }

2)求和

#include "stdafx.h" #include <iostream> #include <string> #include <stdarg.h> using namespace std; int sum(int num,...) { int result=0; va_list arg_ptr; va_start(arg_ptr,num); while (num--) { result+=va_arg(arg_ptr,int); } va_end(arg_ptr); return result; } int main(int argc, char* argv[]) { int result; cout<<"sum of 1,2,3,4,5,6"<<endl; result=sum(6,1,2,3,4,5,6); cout<<result<<endl; cout<<"sum of 1,2,3"<<endl; result=sum(3,1,2,3); cout<<result<<endl; return 0; }

3)使用vsprintf将可变参数全部写入缓冲区

函数声明:int   vsprintf(char   *buf,const   char   *format,   va_list   arglist);    
函数用途:该函数作用同sprintf函数,区别是参数表由一个va_list类型的指针代替  
头文件  stdio.h   stdarg.h  
下例摘自S3C2440中的串口打印函数

void Uart_SendByte(int data) { if(whichUart==0) { if(data=='/n') { while(!(rUTRSTAT0 & 0x2)); // Delay(1); //because the slow response of hyper_terminal WrUTXH0('/r'); } while(!(rUTRSTAT0 & 0x2)); //Wait until THR is empty. // Delay(1); WrUTXH0(data); } else if(whichUart==1) { if(data=='/n') { while(!(rUTRSTAT1 & 0x2)); //Delay(1); //because the slow response of hyper_terminal rUTXH1 = '/r'; } while(!(rUTRSTAT1 & 0x2)); //Wait until THR is empty. //Delay(1); rUTXH1 = data; } else if(whichUart==2) { if(data=='/n') { while(!(rUTRSTAT2 & 0x2)); //Delay(1); //because the slow response of hyper_terminal rUTXH2 = '/r'; } while(!(rUTRSTAT2 & 0x2)); //Wait until THR is empty. //Delay(1); rUTXH2 = data; } } void Uart_SendString(char *pt) { while(*pt) Uart_SendByte(*pt++); } void Uart_Printf(char *fmt,...) { va_list ap; char string[256]; va_start(ap,fmt); vsprintf(string,fmt,ap); Uart_SendString(string); va_end(ap); }

 

你可能感兴趣的:(C语言中可变参数函数的使用)