为什么对于继承虚函数的类,boost :: optional失败

时间:2011-06-22 00:40:08

标签: c++ gcc boost gcc-warning strict-aliasing

升压::可选<>适用于简单数据类型,但只要用于从实现接口的类继承的类,它就会在启用严格别名时失败。

示例:

#include <boost/optional.hpp>

struct MyLine{
  double a;
  double b;
};

class Edge{
  public:
    MyLine toMyLine() const;
  private:
    virtual MyLine doToMyLine() const =0;
};

class Wall:public Edge {
  public:
    Wall(MyLine const& seg):mMyLine(seg){};
  private:
    MyLine doToMyLine() const{return MyLine();};
    MyLine mMyLine;
};

class SimpleWall {
  public:
    SimpleWall(MyLine const& seg):mMyLine(seg){};
  private:
    MyLine mMyLine;
};

int main(){
 //boost::optional<Wall> res;       //fails with strict aliasing error
 boost::optional<SimpleWall> res2; //compiles just fine
}

使用gcc版本4.4.3编译以下内容时出错:

g++ -c -pipe -Wall -Wextra -Wunused -Wmissing-declarations -Wpointer-arith -Wcast-align -Wwrite-strings -Wredundant-decls -Werror -std=c++0x -O2 -Wall -W -I/usr/local/boost_1_44_0 -o obj/main.o main.cpp

解决此问题的最佳方法是什么?我非常希望启用严格别名警告。我使用的是升级版本1.44。

更新:

变得更糟!!请考虑以下代码:

#include <boost/optional.hpp>

class MyBase{
  public:
    int toFoo() const;
  private:
    virtual int doToFoo() const =0;
};

class Child:public MyBase {
  public:
    Child(int const& foo):mFoo(foo){};
  private:
    int  doToFoo() const{return 0;}
    int mFoo;
};

int main(){
 boost::optional<int> optint;       //comment out for surprise
 optint.get();                      //comment out for surprise
 boost::optional<Child> res2;
 res2.get();
}

使用gcc版本4.4.3编译以下内容,编译:

g++ -c -pipe -Wall -Wextra -Wunused -Wmissing-declarations -Wpointer-arith -Wcast-align -Wwrite-strings -Wredundant-decls -Werror -std=c++0x -O2 -Wall -W -I/usr/local/boost_1_44_0 -o obj/main.o main.cpp

如果标有“// comment out for surprise”的行已注释掉,我会收到严格的别名警告。我已经检查了至少20次。这是我见过的最奇怪的事情之一。看起来像boost :: optional初始化某事。独立于其模板参数或像gcc一样,只有在用sth调用时才能理解boost :: optional。琐碎的。有什么想法吗?

1 个答案:

答案 0 :(得分:4)

我在Boost 1.44.0中尝试了该程序。 这个问题的原因是不要覆盖doToSegment。

Segment doToSegment(){};

应该添加const:

Segment doToSegment() const {};