1.printf函数与可变参数列表
typedef char* va_list; va_start(ap, A) { char* ap = ((char*)(&A)) + sizeof(A); } int printf(char* fmt, . . .) { va_list args; int n; va_start(args, fmt); n = vsprintf(sprint_buf, fmt, args); va_end(args); write(stdout, sprint_buf, n ); return n; }
printf通过va_start将所有可变参数放到了由args指向的指向的一块内存中,再调用vsptrintf,真正参数的格式以及个数是在vsprintf中搞定的。这里主要关注va_start函数。
调用va_start(args, fmt)函数即将args执行fmt偏移sizeof(fmt)大小的位置,正好是fmt后面第一个参数的地址。而函数栈中参数根据从右往左入栈的,知道第一个参数和后面参数的类型就可以得到后面的参数。