一、从汇编角度分析
从汇编角度来看,函数的参数总是从高地址压到低地址,而访问参数的时候又是通过基址加偏移量来的,所以按照逻辑,偏移量为0对应第一个参数,第一个参数在低地址,低地址最后压入栈,相对应的函数最右边的参数也就最先计算,并先压入栈。 由于栈是后进先出所以读取参数时是从左到右。
二、printf参数计算是从右往左
上代码感受一下。
#include<stdio.h>
int main()
{
int a[5]={6,7,8,9,10};
int *p=a;
printf("%d,%d\n",*p,*++p);
return 0;
}
可以看到输出结果是:7,7
而不是:6,7
三、类似的所有函数的参数计算都是从右至左
类似printf的所有函数,包括算定义函数的参数计算都是从右至左。函数数有多个参数时计算总得有个顺序吧?不是从左至右,就是从右至左,抑或从中间向两边;一句话选定一个顺序后就“大家都这么办”,总不能有些函数从左至右,有些函数从右至左吧,那编译器就太难做了。当初选择从右至左肯定是这样有好多方便之处,比如printf中的参数表,由于C是基于栈操作的,栈又是后进先出的,从右至左计算压栈,然后按弹出顺序输出到屏幕上刚好顺应了大多数文本从左至右的习惯,很是方便;若从左至右计算压栈,那也不是说不可以,但处理就没有这么方便了。
代码演示:
#include<stdio.h>
void zaizai(int a,int b,int c)
{
printf("%d,%d,%d", a, b, c);
}
int main()
{
int a = 3;
zaizai(a++, a++, a++);
return 0;
}
输出结果是:5,4,3
总结:
从上面分析可以看出,函数参数和逗号运算符的执行顺序正好相反!