3

尝试编译以下代码时,我收到了一个非常奇怪的 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)
4

3 回答 3

6

gcc 4.4 有一个错误,其中 std::map错误地中断 警告有关严格别名规则的错误。

http://gcc.gnu.org/bugzilla/show_bug.cgi?id=39390

您的代码是有效的 C++。严格的别名仅允许使用时默认启用的优化子集-O3

您的解决方案是使用-fno-strict-aliasing或不同版本的 gcc 进行编译。

如果您对什么是严格混叠感到好奇,请参阅此处

于 2009-12-26T22:49:10.890 回答
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 下的版本进行了编译,一切正常。

于 2009-12-26T22:08:53.020 回答
0

哪个版本的g++?g++ 4.3.2 毫无怨言地编译了它。

于 2009-12-26T22:05:54.767 回答