++和--对应的是两头汇编指令:
前置:变量自增(自减)1,取变量值;
后置:取变量值,变量自增(自减)1;
观察下面代码,判断输出:
#include <stdio.h>
int main()
{
int i=0;
int j=0;
j = (i++) + (i++) + (i++);
printf("i = %d\n", i);
printf("j = %d\n\n", j);
i=0;
j=0;
j = (++i) + (++i) + (++i);
printf("i = %d\n", i);
printf("j = %d\n", j);
return 0;
}
在Linux下使用GCC进行编译运行:
i = 3
j = 0
i = 3
j = 7
为什么会出现这种情况呢?
查看汇编代码可发现:
VC编译器先对i进行三次自增1,然后相加,结果为9;
GCC编译器先对i进行两次自增,相加后再进行自增。
C语言中之规定了++和--对应指令的相对执行次序,所以++和--对应的汇编指令不一定连续执行。
在混合运算中,++和--的汇编指令可能会被打断。
所以,必须注意的是,++和--在混合运算中,结果是不确定的。在工程中应该避免这种写法。
笔试面试中的“奇葩”题:
1、++i+++i+++i;
2、a+++b;
首先先了解一下“贪心算法”:含++和--表达式的阅读技巧
编译器处理的每个符号一应该尽可能多的包含字符;
编译器从左往右的顺序一个一个尽可能多的读入字符;
当读入的字符不可能和已读入的字符组成合法符号为止。
int main()
{
int i=0;
int j = ++i+ ++i+ ++i; // int j= ++i+++i+++i; -->>error
printf("j = %d\n", j);
int a=1;
int b=4;
int c = a+++b;
int* p =&a;
b = b/ *p; //b = b/*p; -->> error
printf("a = %d\n", a);
printf("b = %d\n", b);
printf("c = %d\n", c);
return 0;
}
为了避免编译器的“贪心”,可以使用空格作为一个运算符的休止符,编译器在读入空格之后会立即对之前读入的符号进行处理。所以在日常的代码编写中,建议可以加空格的地方不要省略,增强程序的可读性。
需要注意的是:
int a = 1;
int b = 2;
int m = ((a++) < (b) ? (a++) : (b));
程序运行后,m=2, a=3。不会发生短路。
小结:
++和--操作符在混合运算中的行为可能不同。
编译器通过贪心算法处理表达式中的子表达式;
空格可以作为C语言中的一个完整符号的休止符;
编译器读入空格之后立即对之前的符号进行处理。