资源描述:
《printf()函数实现机制》由会员上传分享,免费在线阅读,更多相关内容在教育资源-天天文库。
1、printf()函数实现机制printf()函数调用的一般格式:printf("<格式化字符串>",<参量表>);一般字符格式化规定字符可有可无printf()函数调用的一般格式:printf("<格式化字符串>",<参量表>);可变参数列表printf实例1:#includeintmain(){printf(“HelloWorld!”);return0;}传递一个参数!printf实例2:#includeintmain(){inti=3;printf(“%d”,i);return0;}传递了两
2、个参数!printf实例3:#includeintmain(){inti=3;printf(“%d%d%d%d”);return0;}printf实例4:#includeintmain(){inti=3,j=4;printf(“%d”,i,j);return0;}问题:1.怎样让printf()函数知道传递了多少参数?2.printf()函数怎样访问这些参数?printf()在VC6.0中的库中的源代码:int__cdeclprintf(constchar*format,...){va_li
3、stap;intbuffing;intretval;va_start(ap,format);_ASSERTE(format!=NULL);_lock_str2(1,stdout);buffing=_stbuf(stdout);retval=_output(stdout,format,ap);_ftbuf(buffing,stdout);_unlock_str2(1,stdout);return(retval);}●int__cdeclprintf(constchar*format,...)_cdecl是C和C++程序的缺省调用方式_cdecl调用约定
4、:1.参数从右到左依次入栈2.调用者负责清理堆栈3.参数的数量类型不会导致编译阶段的错误●(constchar*format,...)在形参列表里有这么一个“...”,这个是可变形参的一种写法。当传递参数的个数不确定时,就可以用这种方式来表示。●以下为VC++中stdarg.h里x86平台的宏定义typedefchar* va_list;#defineva_start(va_listap,format)(ap=(va_list)&format+_INTSIZEOF(format))#defineva_arg(va_listap,type)(*(type*)
5、((ap+=_INTSIZEOF(type))-_INTSIZEOF(type)))#defineva_end(va_listap) (ap=(va_list)0)va_start:为获取可变数目参数的函数的参数提供一种便捷手段。设置ap为指向传给函数参数列表中的第一个可选参数的指针,且该参数必须是va_list类型。format是在参数列表中第一个可选参数前的必选参数。va_arg:返回由ap所指向的参数的值,且自增指向下一个参数的地址。type为当前参数的类型,用来计算该参数的长度,确定下一个参数的起始位置。它可以在函数中应用多次,直到得到函数的
6、所有参数为止,但必须在宏va_start后面调用。va_end:在获取所有的参数后,设置指针ap为null。●栈示意图…………参数n参数n-1Printf(参数1,参数2,……,参数n-1,参数n,……)……参数2参数1低地址高地址Format指向参数1执行完va_start后,ap指向参数2(即第一个可变参量)执行完va_arg后,ap指向下一个参数printf()在VC6.0中的库中的源代码:int__cdeclprintf(constchar*format,...){va_listap;intbuffing;intretval;va_start(ap
7、,format);_ASSERTE(format!=NULL);_lock_str2(1,stdout);buffing=_stbuf(stdout);retval=_output(stdout,format,ap);_ftbuf(buffing,stdout);_unlock_str2(1,stdout);return(retval);}retval=_output(stdout,format,ap);●_output(stdout,format,ap)在VC中,_output的定义相当复杂,光是变量声明就有20多行,主体是一个大的循环体。●_out
8、put(stdout,format,a