考虑以下简单的程序(改编自这个问题):
#include <cstdlib>
int main(int argc, char** argv) {
int mul1[10] = { 4, 1, 8, 6, 3, 2, 5, 8, 6, 7 }; // sum = 50
int mul2[10] = { 4, 1, 8, 6, 7, 9, 5, 1, 2, 3 }; // sum = 46
int x1 = std::atoi(argv[1]);
int x2 = std::atoi(argv[2]);
int result = 0;
// For each element in mul1/mul2, accumulate the product with x1/x2 in result
for (int i = 0; i < 10; ++i) {
result += x1 * mul1[i] + x2 * mul2[i];
}
return result;
}
我相信它在功能上等同于以下一个:
#include <cstdlib>
int main(int argc, char** argv) {
int x1 = std::atoi(argv[1]);
int x2 = std::atoi(argv[2]);
return x1 * 50 + x2 * 46;
}
然而clang 3.7.1、gcc 5.3和icc 13.0.1似乎无法进行这样的优化,即使使用-Ofast
. (注意生成的程序集在编译器之间是如何大不相同的!)。但是,当从等式中删除mul2
和时, clang能够执行类似的优化,即使是.x2
-O2
是什么阻止了两个编译器将第一个程序优化为第二个程序?