C++/Qt - QGraphicsItem 的多重继承没有按预期工作

标签 c++ qt multiple-inheritance

我最近在我的小程序中遇到了一个奇怪的问题,如果你能帮助我找出这种行为的原因,那就太好了。

我的任务非常简单 - 我想使用 Qt 图形框架来显示一些对象,我希望 Box2D 计算物体的位置。所以我的类层次结构如下所示:

我有 1 个基本抽象类 B2DObject。它包含一些 Box2D 人员及其后继者的一些通用参数(名称、一些标志等)。它还具有几个将在后续类中重新实现的纯虚函数。

然后我实现了一些表示基本形状的类:圆形、矩形、多边形等。我是通过以下方式实现的:

class ExtendedPolygon : public B2DObject, public QGraphicsPolygonItem { ... };
class ExtendedCircle : public B2DObject, public QGraphicsEllipseItem { ... };

等 (对于那些不熟悉Qt的人,QGraphics***Item继承自QGraphicsItem)。

另外我继承了QGraphicsScene并重新实现了它的mousePressEvent。在此函数中,我使用 QGraphicsScene::itemAt 函数(返回 QGraphicsItem*)请求放置在屏幕上某个点的对象,将其转换为 B2DObject* 并尝试从该对象获取一些内部字段:

void TestScene::mousePressEvent (QGraphicsSceneMouseEvent *event)
{
    QGraphicsItem* item = itemAt (event->scenePos ());
    if (item)
    {
        B2DObject* obj = reinterpret_cast < B2DObject* > (item);
        QString objName = obj->Name();  // just for example, 
                                        // getting other internal fields has 
                                        // the same effect (described below)
        // use retrieved field somehow (e.g. print in the screen)
    }

    // give the event to the ancestor
}

不幸的是,dynamic_cast 在这里不起作用,因为这些类完全不相关。

然后我创建必要的对象并将其添加到我的场景中:

ExtendedPolygon* polygon = new ExtendedPolygon (parameters);
polygon->setName (QString ("Object 1"));
...
TestScene scene;
scene.addItem (polygon);

(对于Qt不熟悉的 friend ,这里是最后一个函数的原型(prototype):

void QGraphicsScene::addItem(QGraphicsItem *item);

我猜它只是将所有项目存储在内部索引存储中,并在项目需要重新绘制时调用 QGraphicsItem::paint (...)。我想 QGraphicsScene 没有对此项进行任何重大更改)。

因此,当我运行程序并单击屏幕上的某个项目时,我的问题就开始了。 TestScene::mousePressEvent 被调用(见上面的一段代码)。

检索鼠标点击位置,找到项目。转换工作正常:在调试器窗口中(我正在使用 Qt Creator)我看到 obj 指向 ExtendedPolygon(地址与我将项目添加到场景时的地址相同,在调试器窗口中我可以看到所有字段)。但是当我得到一些字段时,无论如何我都会收到垃圾(这并不重要,我试图得到什么——一个 QString 或一个指向其他结构的指针)。

所以首先,我想获得有关我的多重继承的任何建议。在 95% 的情况下我都尽量避免它,但从编程的角度来看它在这里非常有效。因此,如果您向我提供您对类层次结构体系结构的观点,我将不胜感激 - 它是否能像我预期的那样工作?

如果在这个层面上一切都很好,那么如果有人知道为什么它不起作用,那就太好了。

我有一些关于解决方法的想法,但我真的很想解决这个问题(只是为了不再重复同样的错误)。

最佳答案

看来我已经找到问题的根本原因了。只是缺乏关于多重继承如何在数据层上真正起作用的知识。

假设我们有 2 个基本类,A 和 B。它们每个都提供一些内部数据字段和一些接口(interface)。

然后我们创建一个派生类AABB,同时继承A和B:

class AABB : public A, public B {...}

AABB 可以添加一些额外的数据字段并重新实现一些接口(interface),但这不是必需的。

让我们创建 AABB 类的对象:

AABB* obj = new AABB ();

例如,obj 指向地址 0x8416e0。在此地址开始来自祖先类 A 的数据。来自祖先类 B 的数据以某个偏移量开始(它应该 bw 等于 sizeof (A)),例如,在 0x841700。

如果我们有一些函数 f (B* b),并且如果我们将指向 AABB 对象的指针传递给该函数(像这样:f (obj)obj 是上面创建的),实际上传递的不是obj起始地址,而是AABB对象B数据段开始处的指针。

因此,这种对多重继承内部工作的误解导致了我遇到的问题。

关于C++/Qt - QGraphicsItem 的多重继承没有按预期工作,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/13428051/

相关文章:

c++ 自动覆盖有符号/无符号

java - 绕过多重继承限制的正确方法是什么?

c++ - 运行 Casablanca 程序失败,在 Mac OS 上找不到 'openssl/conf.h' 文件

c++ - 使用 malloc 分配比现有内存更多的内存

c++ - 在带有 QPushButton 的 Qt 中使用 setWhatIsThis

qt - ListView positionViewAtIndex()不工作

templates - 具有特征和函数重载的静态类型语言?

c++ - 多重继承引起的C++拷贝构造函数调用不明确

c++ - 无法用visual studio 2017编译静态库项目

c++ - 如何访问由另一个 API 定义和分配的变量