如果某个操作符的各个操作数属于不同的类型,那么除非其中一个操作数转换为另一个操作数的类型否则操作就无法进行。
寻常算术转换:
1.long double
2.double
3.float
4.unsigned long int
5.long int
6.unsigned int
7.int
如果某个操作数的类型在这个列表中排名靠后,那么首先要转换为另一个操作数的类型后执行运算
例如:int a;
double b;
如果要进行a+b,那么首先要把a转换为double类型
问题表达式解析
表达式1:
a*b+c*d+e*f
可能为a*b->c*d->e*f->第一个+->第二个+
或者为a*b->c*d->第一个+->e*f->第二个+
解决办法:1.加括号2.拆开写
表达式2:
c+--c
假设初始值 int c = 5;,可能出现两种典型结果:
- 情况 1:编译器先计算右侧
--c(先修改c),再计算左侧c- 步骤:
--c执行,c变为 4(副作用完成),然后左侧c读取的值是 4,最终结果4+4=8
- 步骤:
- 情况 2:编译器先计算左侧
c(先读取原始值),再计算右侧--c(后修改c)- 步骤:左侧先读取
c的原始值 5,然后执行--c使c变为 4,最终结果5+4=9
- 步骤:左侧先读取
- 情况 3:更极端的优化下,可能得到其他无意义结果(标准不约束,编译器可自由处理)
表达式3:
cpp
#include<stdio.h>
int fun()
{
static int count = 1;
return ++count;
}
int main()
{
int answer;
answer = fun() - fun() * fun();
return 0;
}
因为fun函数的调用顺序不固定,所以answer结果也会不同
cpp
#include<stdio.h>
int main()
{
int i = 1;
int ret = (++i) + (++i) + (++i);
return 0;
}
在执行第一个+时最后一个(++i)有没有执行不确定
即使有了操作符的优先级和结合性,写出的表达式依然可能不能通过操作符的属性,确定唯一的计算路径,那这个表达式就是存在潜在风险的。