Ошибка проверки времени выполнения # 0 - значение ESP не было должным образом сохранено при вызове функции - PullRequest
7 голосов
/ 24 декабря 2011

Я создал простую программу, которая демонстрирует ошибку времени выполнения, которую я получаю с моим приложением 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.

Есть идеи, в чем проблема?

Ответы [ 2 ]

8 голосов
/ 24 декабря 2011

Поскольку вы используете множественное наследование, указатель vftable на то, что, как ожидается, будет MyInterface*, фактически является указателем на QGraphicsLineItem vftable.

A 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)
1 голос
/ 24 декабря 2011

Ваша проблема будет исправлена, если вы используете динамическое приведение

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

Этот вопрос касается различных приведений C ++ и когда их использовать.В вашем случае из-за множественного наследования вы должны использовать динамическое приведение

...