C++ Создание экземпляров потомков абстрактного класса с чисто виртуальными функциями

У меня есть абстрактный класс, пусть это будет, например, Animal. У животного есть чисто виртуальная функция еды, которую каждое животное должно выполнять, если оно не хочет голодать. Я гарантирую, что таким образом может быть создан только ребенок Животного:

Animal.hpp

class Animal
{

public:

    enum eAnimal{
        CAT=0,
        DOG=1,
        BIRD=2
    };

    // Instantiates the desired animal.
    static Animal GetAnimal(const int animal);

    virtual void Eat() const = 0;

protected:

    Animal();
};

Animal.cpp

Animal Animal::GetAnimal(const int animal)
{
    switch(animal)
    {
    case CAT:
        return Cat();
    case DOG:
        return Dog();
    case BIRD:
        return Bird();
    default:
        cerr << "Animal not recognized." << endl;
        exit(-1);
    }
}

Animal::Animal()
{}

При этом кот будет:

Cat.hpp

class Cat : public Animal
{

public:

    Cat();

    void Eat() const;
};

Cat.cpp

Cat::Cat() : Animal()
{}

void Cat::Eat() const
{
    // The cat eats.
}

Однако этот код не работает, он получает ошибку недопустимого абстрактного возвращаемого типа "Animal" в GetAnimal, потому что Animal является абстрактным и не может быть создан, хотя мой API гарантирует, что это не будет.

Какие умные решения могут иметь эту проблему? Я могу сделать функцию Eat не чистой и дать ей реализацию по умолчанию, но я бы не хотел этого делать.

3 ответа

Решение
Animal Animal::GetAnimal(const int animal)
{
    switch(animal)
    {
    case CAT:
        return Cat();
    case DOG:
        return Dog();
    case BIRD:
        return Bird();
    default:
        cerr << "Animal not recognized." << endl;
        exit(-1);
    }
}

Вы говорите, что GetAnimal должен вернуть Animal объект, это не то, как наследование работает, наследование работает в основном через указатели. Когда вы пытаетесь вернуть объект типа, компилятор неявно должен создать Animal объект, но это не разрешено, потому что Animal это абстрактный класс. Даже если бы вы должны были сделать eat() только virtual у вас все еще будет проблема с нарезкой объектов.

Вы можете сделать это вернуть Animal* и освободите результат после того, как вы его использовали:

Animal* Animal::GetAnimal(const int animal)
{
    switch(animal)
    {
    case CAT:
        return new Cat();
    case DOG:
        return new Dog();
    case BIRD:
        return new Bird();
    default:
        cerr << "Animal not recognized." << endl;
        exit(-1);
    }
}

Абонент:

Dog* myDog = GetAnimal(DOG);

//somewhere later when you dont need myDog anymore..(don't forget this!)
delete myDog;

Но если у вас есть доступ к C++11 или более поздней версии, я рекомендую использовать умные указатели вместо необработанных указателей, чтобы указатель освободился сам по себе:

#include <memory>
std::unique_ptr<Animal> Animal::GetAnimal(const int animal)
{
    switch(animal)
    {
    case CAT:
        return std::make_unique<Cat>();
    case DOG:
        return std::make_unique<Dog>();
    case BIRD:
        return std::make_unique<Bird>();
    default:
        cerr << "Animal not recognized." << endl;
        exit(-1);
    }
}

Абонент:

auto dog = GetAnimal(DOG);
//no explicit delete required.

Вы хотите, чтобы возвращаемый тип GetAnimal был указателем на Animal, а не на само Animal. В общем, всякий раз, когда вы пытаетесь использовать полиморфизм, указатели - лучший способ.

Мое лучшее предположение, что это то, что происходит: вы создаете экземпляр Cat. Затем в вашем операторе возврата, поскольку вы не возвращаете указатель, он должен сделать копию ваших данных. Поскольку тип возвращаемого значения - Animal, он пытается вызвать конструктор копирования по умолчанию для Animal. Поэтому он пытается создать экземпляр класса Animal, что, конечно, невозможно.

Виртуальные функции-члены работают с указателями. Просто измените немного свой API на что-то вроде этого:

std::unique_ptr<Animal> Animal::GetAnimal(const int animal)
{
    switch(animal)
    {
    case CAT:
        return std::make_unique<Cat>();
    (...)
    }
}

Обратите внимание, что я предполагаю, что вы используете, по крайней мере, C++14

Другие вопросы по тегам