运行时检查失败 #0 - ESP 的值未在函数调用中正确保存

发布于 2024-12-22 21:49:59 字数 1536 浏览 4 评论 0原文

我创建了一个简单的程序,演示了使用多重继承的 Qt 应用程序遇到的运行时错误。继承树如下所示:

QGraphicsItem (abstract)
      \
     QGraphicsLineItem      MyInterface (abstract)
                 \          /
                  \        /
                  MySubclass

这是代码:

/* main.cpp */
#include <QApplication>
#include <QGraphicsScene>
#include <QGraphicsLineItem>

//simple interface with one pure virtual method
class MyInterface
{
public:
  virtual void myVirtualMethod() = 0;
};

//Multiple inheritance subclass, simply overrides the interface method
class MySubclass: public QGraphicsLineItem, public MyInterface
{
public:
  virtual void myVirtualMethod() { }
};

int main(int argc, char** argv)
{
  QApplication app(argc, argv); //init QApplication
  QGraphicsScene *scene = new QGraphicsScene(); //create scene

  scene->addItem(new MySubclass()); // add my subclass to the scene

  Q_FOREACH(QGraphicsItem *item, scene->items()) // should only have one item
  {
    MyInterface *mInterface = (MyInterface*)item; // cast as MyInterface
    mInterface->myVirtualMethod(); // <-- this causes the error
  }
  return 0;
}

当调用我的接口方法时,在 Visual Studio 中进行调试会导致运行时错误:

    Run-Time Check Failure #0 - The value of ESP was not properly 
    saved across a function call.  This is usually a result of 
    calling a function declared with one calling convention with 
    a function pointer declared with a different calling convention.

知道问题是什么吗?

I created a simple program that demonstrates the runtime error I'm getting with my Qt application that uses multiple inheritance. The inheritance tree looks like:

QGraphicsItem (abstract)
      \
     QGraphicsLineItem      MyInterface (abstract)
                 \          /
                  \        /
                  MySubclass

And here is the code:

/* main.cpp */
#include <QApplication>
#include <QGraphicsScene>
#include <QGraphicsLineItem>

//simple interface with one pure virtual method
class MyInterface
{
public:
  virtual void myVirtualMethod() = 0;
};

//Multiple inheritance subclass, simply overrides the interface method
class MySubclass: public QGraphicsLineItem, public MyInterface
{
public:
  virtual void myVirtualMethod() { }
};

int main(int argc, char** argv)
{
  QApplication app(argc, argv); //init QApplication
  QGraphicsScene *scene = new QGraphicsScene(); //create scene

  scene->addItem(new MySubclass()); // add my subclass to the scene

  Q_FOREACH(QGraphicsItem *item, scene->items()) // should only have one item
  {
    MyInterface *mInterface = (MyInterface*)item; // cast as MyInterface
    mInterface->myVirtualMethod(); // <-- this causes the error
  }
  return 0;
}

Debugging in visual studio results in a runtime error when my interface method is called:

    Run-Time Check Failure #0 - The value of ESP was not properly 
    saved across a function call.  This is usually a result of 
    calling a function declared with one calling convention with 
    a function pointer declared with a different calling convention.

Any idea what the problem is?

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。

评论(2

初熏 2024-12-29 21:49:59

由于您使用的是多重继承,因此指向 MyInterface*vftable 指针实际上是指向 QGraphicsLineItem vftable 的指针。

dynamic_cast 可以解决这个问题,因为它将返回正确的 vftable

MyInterface* mInterface = dynamic_cast<MyInterface*>(item);

一个简单的例子:

class A
{
public:
    virtual void foo() = 0;
};

class B
{
public:
    virtual void goo() {};
};

class C : public B, public A
{
public:
    virtual void foo() {}; 
};

//....

B* c = new C;                  // c  is at 0x00a97c78 
A* a = (A*)c;                  // a  is at 0x00a97c78 (vftable pointer of B) 
A* a1 = dynamic_cast<A*>(c);   // a1 is at 0x00a97c7c (vftable pointer of A)

Because you're using multiple inheritance, the vftable pointer to what is expected to be a MyInterface* is actually a pointer to a QGraphicsLineItem vftable.

A dynamic_cast would solve the issue because it will return the correct vftable

MyInterface* mInterface = dynamic_cast<MyInterface*>(item);

A simple example:

class A
{
public:
    virtual void foo() = 0;
};

class B
{
public:
    virtual void goo() {};
};

class C : public B, public A
{
public:
    virtual void foo() {}; 
};

//....

B* c = new C;                  // c  is at 0x00a97c78 
A* a = (A*)c;                  // a  is at 0x00a97c78 (vftable pointer of B) 
A* a1 = dynamic_cast<A*>(c);   // a1 is at 0x00a97c7c (vftable pointer of A)
梦里的微风 2024-12-29 21:49:59

如果您使用动态转换

MyInterface* mInterface = dynamic_cast<MyInterface*>(item);

这个问题涉及各种 C++ 类型转换以及何时使用它们。在您的情况下,由于多重继承,您应该使用动态强制转换

You issue will be fixed if you use a dynamic cast

MyInterface* mInterface = dynamic_cast<MyInterface*>(item);

This questions deals various C++ casts and when to use them. In your case because of multiple inheritance you should be using dynamic casts

~没有更多了~
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文