Выделение объекта абстрактного класса типа 'CObject' - PullRequest
0 голосов
/ 26 апреля 2018

Я хочу реализовать CWindow и другие элементы, используя полиморфизм и абстрактный класс CObject. Но я получил ошибку «Выделение объекта абстрактного класса типа« CObject ».

    class CObject
{
public:
    CObject ( const int id, const string &title, const CRect &pos ):
    m_id(id),m_title(title),m_pos(make_unique<CRect>(pos))
    {
    }
    virtual void print ( ostream &os ) const = 0;
    virtual void add ( const CObject &src ){            
        m_src.push_back(make_unique<CObject>(src)); //->here the problem
    }
    friend ostream & operator << (ostream &os, const CObject &src ){
        src.print(os);
        return os;
    }
protected:
    int m_id;
    string m_title;
    unique_ptr<CRect> m_pos;
    vector< unique_ptr<CObject> > m_src;
};

Например, я хочу создать кнопку, а затем добавить кнопку в CWindow (в основной функции). Окно имеет абсолютную позицию, но другие части окна зависят от CWindow. Затем я хочу изменить размер окна, и все элементы изменится автоматически, поэтому я должен сохранить все данные о подэлементах.

class CWindow : public CObject
    {
    public:
        CWindow ( const string & title, const CRect & absPos ): CObject(0,title,absPos)
        {
        };
        virtual void print ( ostream &os ) const{
            os << "CWindow";
        };
        CWindow & Add ( const CObject &src ){
            add(src);
            return *this;
        }
    };
class CButton : public CObject
{
public:
    CButton ( int  id, const CRect &relPos, const string &name ):
    CObject(id,name,relPos)
    {
    }
};

Пример основной функции

    int main ( void )
    {
         CWindow a ( "Sample window", CRect ( 10, 10, 600, 480 ) );
            a . Add ( CButton ( 1, CRect ( 0.1, 0.8, 0.3, 0.1 ), "Ok" ) ) 
. Add ( CButton ( 2, CRect ( 0.6, 0.8, 0.3, 0.1 ), "Cancel" ) );
    }

Ответы [ 2 ]

0 голосов
/ 26 апреля 2018

CObject - абстрактный класс, потому что print() - абстрактный. Таким образом, вы НЕ МОЖЕТЕ создавать экземпляры CObject напрямую. Вы можете создавать только конкретные производные классы, которые реализуют print().

Вызов std::make_unique<CObject>(...) пытается создать экземпляр CObject, поэтому в вашем коде происходит сбой. Вместо этого вам нужно указать производный класс, например: std::make_unique<CButton>(...).

Однако показанный вами метод add() не будет работать по ряду причин:

  • нет способа узнать, был ли переданный CObject выделен в автоматической или динамической памяти. Пример main(), который вы показали, создает объекты в автоматической памяти, но для удаления по умолчанию std::unique_ptr (std::delete) вместо этого требуется динамическая память.

  • вызывающий абонент не может указать правильный тип для передачи параметру шаблона T std::make_unique().

  • ни один из ваших производных классов не имеет конструктора, который принимает CObject в качестве входных данных, поэтому вы не можете указать переданный CObject в параметре args std::make_unique().

Вам нужно изменить add(), чтобы вместо него взять std::unique_ptr<CObject>, а затем вы можете просто std::move() ввести его в vector, например:

virtual void add(std::unique_ptr<CObject> src)
{
    m_src.push_back(std::move(src));
}

Пусть вызывающая сторона обрабатывает создание нужного объекта, add() просто становится владельцем этого объекта:

int main()
{
    CWindow a("Sample window", CRect(10, 10, 600, 480));
    a.add(std::make_unique<CButton>(1, CRect(0.1, 0.8, 0.3, 0.1), "Ok"));
    a.add(std::make_unique<CButton>(2, CRect(0.6, 0.8, 0.3, 0.1), "Cancel"));
}

В противном случае, если вы хотите, чтобы add() позвонил std::make_unique() от имени вызывающего абонента, вам понадобится что-то более похожее на это:

template<class T, class... Args>
void add(Args&&... args)
{
    m_src.push_back(std::make_unique<T>(args));
}

...

int main()
{
    CWindow a("Sample window", CRect(10, 10, 600, 480));
    a.add<CButton>(1, CRect(0.1, 0.8, 0.3, 0.1), "Ok");
    a.add<CButton>(2, CRect(0.6, 0.8, 0.3, 0.1), "Cancel");
}
0 голосов
/ 26 апреля 2018

Вы должны реализовать метод print в классе CObject. Eg.:

virtual void print ( ostream &os ) const {};
...