问题如下:
1:为什么这样的代码为报错
cpp
#define _CRT_SECURE_NO_WARNINGS
#include <iostream>
#include <algorithm>
#include <cstdio>
#include<string>
#include<stdlib.h>
using namespace std;
int main()
{
int i = 7; printf("%d\n", i++ * i++);//49
int a[10] = {0};
for (i = 0; i < 10; i++)
{
cout << a[i] << endl;
}
a[i] = i++;
printf("%d",a[i]);
return 0;
}
通过调试发现问题出在for循环后i此时为10,a[10]就已经数组越界了,因此a[i]是a[10]不是a[7],打印的a[i]是错误值
2: 下面的代码 int i=7; printf("%d\n", i++ * i++); 返回 49?不管按什么顺序计算, 难道不该打印出56吗?
cpp
#define _CRT_SECURE_NO_WARNINGS
#include <iostream>
#include <algorithm>
#include <cstdio>
#include<string>
#include<stdlib.h>
int main()
{
int i = 7; printf("%d\n", i++ * i++);//49
return 0;
}
问题解释:
参考书籍《你必须知道的495个C语言问题》
后置++和后置--是先使用再自增自减,没有任何保证确保自增或自减会在输出变量原值之 后和对表达式的其它部分进行计算之前立即进行。也不能保证变量的更新会在表 达式 "完成") 之前 的某个时刻进行。本例中, 编译器选择使用变量的旧值相乘以后再对二者进行自 增运算。
包含多个不确定的副作用的代码的行为总是被认为未定义。(简单而言, "多 个不确定副作用" 是指在同一个表达式中使用导致同一对象修改两次或修改以后 又被引用的自增, 自减和赋值操作符的任何组合。
我们不需要探究这些在编译器中是如何实现的,但要知道后置++,后置--与前置++,前置--的区别。
3:对于代码 int i = 3; i = i++; 不同编译器给出不同的结果, 有的为 3, 有的为 4, 哪个是正确的?
在VS2022编译器上结果为4
但不一定每个编译器都是4。i++ 和 ++i 都不同于 i+1。如果你要使 i 自增 1, 使用 i=i+1, i+=1, i++ 或 ++i, 而不是任何组合。
4:我可否用括号来强制执行我所需要的计算顺序?
cpp
#define _CRT_SECURE_NO_WARNINGS
#include <iostream>
#include <algorithm>
#include <cstdio>
#include<string>
#include<stdlib.h>
int f()
{
return 2;
}
int g()
{
return 3;
}
int h()
{
return 4;
}
int main()
{
int a = f() + g() * h();
printf("%d\n", a);
return 0;
}
我们知道*运算符优于+运算符,但f(),g(),h()函数哪一个先被调用呢,通过调试我发现调用顺序先f(),接着g(),最后h()。如果运算表达式为f()+h()*g()则先f(),再h()最后g()。