在尝试编译以下代码时,我收到了一个非常奇怪的g ++警告:
#include <map>
#include <set>
class A {
public:
int x;
int y;
A(): x(0), y(0) {}
A(int xx, int yy): x(xx), y(yy) {}
bool operator< (const A &a) const {
return (x < a.x || (!(a.x < x) && y < a.y));
}
};
struct B {
std::set<A> data;
};
int
main()
{
std::map<int, B> m;
B b;
b.data.insert(A(1, 1));
b.data.insert(A(1, 2));
b.data.insert(A(2, 1));
m[1] = b;
return 0;
}
输出:
$ g++ -Wall -W -O3 t.cpp -o /tmp/t
t.cpp: In function ‘int main()’:
t.cpp:14: warning: dereferencing pointer ‘__x.52’ does break strict-aliasing rules
t.cpp:14: warning: dereferencing pointer ‘__x.52’ does break strict-aliasing rules
/usr/lib/gcc/i686-redhat-linux/4.4.2/../../../../include/c++/4.4.2/bits/stl_tree.h:525: note: initialized from here
根本没有任何意义。我该如何解读?我没看到发布的代码有什么问题。
忘记指定编译器详细信息:
$ gcc --version
gcc (GCC) 4.4.2 20091027 (Red Hat 4.4.2-7)
答案 0 :(得分:6)
gcc 4.4有一个错误,std :: map 中断 错误地警告严格别名规则。
http://gcc.gnu.org/bugzilla/show_bug.cgi?id=39390
您的代码是有效的C ++。严格别名仅允许使用-O3
时默认启用的优化子集。
您的解决方案是使用-fno-strict-aliasing
或不同版本的gcc进行编译。
如果您对严格别名是什么感到好奇,that has been asked here。
答案 1 :(得分:1)
尝试改变这一点:
return (x < a.x || (!(a.x < x) && y < a.y));
成:
return (x < a.x || (a.x == x && y < a.y));
我也使用您的版本和我的版本在g ++ 3.4.2下编译了这一切,一切正常。
答案 2 :(得分:0)
哪个版本的g ++? g ++ 4.3.2无怨无悔地编译它。