在派生类C++中避免"纯虚函数调用"

Den*_*ger 3 c++ virtual inheritance

我是C++的新手,所以如果这个问题的水平略低于通常的标准,我还是要道歉 -

我正在尝试从具有虚函数定义的基类继承几个类,然后我想创建一个MainClass*数组,它可以包含所有派生类,以便输出派生+定义的虚拟功能.

我收到错误"R6025:纯虚函数调用" - 我不知道为什么,我假设 - 当被调用时 - 使用子类中该函数的更新定义.

这是我的代码:

基础:

class mitarbeiter
{
public:
    mitarbeiter(string name, int nummer);
    ~mitarbeiter(void);
    virtual void einkommen() = 0;
protected:
    string name;
    int nummer;

};
Run Code Online (Sandbox Code Playgroud)

Derived1:

#include "mitarbeiter.h"
class lohnempfaenger : public mitarbeiter
{
public:
    lohnempfaenger(int stundenlohn, int anzahlStunden, string name, int nummer);
    ~lohnempfaenger(void);
    void einkommen();
private:
    int stundenlohn;
    int anzahlStunden;
};
Run Code Online (Sandbox Code Playgroud)

派生2:

#include "mitarbeiter.h"
#include <string>
class angestellter : public mitarbeiter
{
public:
    angestellter(int gehalt, string name, int nummer);
    ~angestellter(void);
    void einkommen();
private:
    int gehalt;
};
Run Code Online (Sandbox Code Playgroud)

衍生1中"einkommen()"的实现:

void lohnempfaenger::einkommen()
{
    cout << "Lohnempfaenger, Name: " << this->name << ", Gesamtlohn: " << this->stundenlohn*this->anzahlStunden << "\n";
}
Run Code Online (Sandbox Code Playgroud)

在Derived 2中实现"einkommen()":

void angestellter::einkommen()
{
    cout << "Angestellter, Name: " << this->name << ", Einkommen: " << this->gehalt << "\n";
}
Run Code Online (Sandbox Code Playgroud)

主方法中的数组声明:

mitarbeiter* mitPtr[5];
mitPtr[0] = &angestellter(/*values...*/);
//error
mitPtr[0]->einkommen();
Run Code Online (Sandbox Code Playgroud)

构造函数:

mitarbeiter::mitarbeiter(string name, int nummer)
{
    this->name = name;
    this->nummer = nummer;
}

angestellter::angestellter(int gehalt, string name, int nummer):mitarbeiter(name, nummer)
{
    this->gehalt = gehalt;
}

lohnempfaenger::lohnempfaenger(int stundenlohn, int anzahlStunden, string name, int nummer):mitarbeiter(name, nummer)
{
    this->stundenlohn = stundenlohn;
    this->anzahlStunden = anzahlStunden;
}
Run Code Online (Sandbox Code Playgroud)

非常感谢!

Mik*_*our 7

现在你已经向我们展示了你如何实际初始化指针,这是问题所在:

mitPtr[0] = &angestellter(/*values...*/);
Run Code Online (Sandbox Code Playgroud)

这指向一个临时对象,该对象在语句结束时被销毁.之后使用指针将给出未定义的行为.

你需要一个非临时变量来指向:

angestellter a(/*values...*/);
mitPtr[0] = &a;
Run Code Online (Sandbox Code Playgroud)

或者可能是动态分配的对象:

mitPtr[0] = new angestellter(...);
Run Code Online (Sandbox Code Playgroud)

在这种情况下,不要忘记删除它; 并且不要忘记给基类一个虚拟析构函数,以便删除它.您可以考虑使用智能指针为您删除它; std::unique_ptr会是理想的.

  • @Dennis为了记录,你得到"纯虚拟调用"的原因是因为当调用析构函数时,它会将类的vtable重置为下一个.因此,当调用`~angestellter`时,它会将类的函数重置为`mitarbeiter`的函数 - 这就是为什么如果你试图在`~angestellter`中调用`einkommen()`就会遇到同样的问题.因此,当您调用`einkommen()`时,您正在访问无效内存,但内存可能仍包含其最后有效内容,该内容是指向基类中纯标记的指针. (3认同)