Qt项目中的前向声明

时间:2017-10-04 04:07:55

标签: c++ qt forward-declaration

我正在尝试将Qt与C ++结合使用。我之前使用QT进行Python编程。

我的简单测试不起作用。这是我的tour.h文件:

#ifndef TOUR_H
#define TOUR_H

#include <QtWidgets/QMainWindow>
#include <QtWidgets/QTableView>

class TourTable;

class Tour : public QMainWindow
{
    Q_OBJECT

public:
    Tour();

/*protected:
    void closeEvent(QCloseEvent *event);

private slots:
    void newCompetitor();
    void remCompetitor();
    void finalReport();
    void openPage();
    void savePage();
    void reloadTitle();*/

private:
    TourTable _table;

};

class QStandardItem;
class QStandardItemModel;

class TourTable : public QTableView
{
    Q_OBJECT

public:
    TourTable();
/*  bool isChanged();
    QString windowName();
    void finalReport();
    void newCompetitor();
    void remCompetitor();
    bool savePage();
    void setChanged(bool value);
    void openPage();    

protected:
    void itemChanged(QStandardItem item);

private:*/
//  bool _secondBetter(p1, p2);

    Tour _parent;
//  QStandardItemModel _model;
//  bool _saved;
//  bool _changed;

};

#endif

我已经在这段代码中评论了几乎所有内容来隔离问题,但我仍然不知道是什么导致了这个问题。我是第一次尝试使用C ++。

错误消息是:

tour.h:28:12: error: field ‘_table’ has incomplete type ‘TourTable’
  TourTable _table;
            ^~~~~~
tour.h:7:7: note: forward declaration of ‘class TourTable’
 class TourTable;

有人可以帮我解决这个问题吗?

2 个答案:

答案 0 :(得分:1)

要清楚,此错误来自您的编译器 - 而不是来自qmake - 并且是语言的限制。但是可以通过创建一个定义父/子行为的类并使您希望该行为从基类继承的任何类来轻松克服它,就像Qt的QObject那样。

QMainWindowQTableView都继承自QObject,因此如果您使用QObject的父/子系统,则此设计可能是多余的。要从子呼叫parent()中查找父母,并从父母中找到子女,您可以拨打children()

答案 1 :(得分:1)

C ++中的前向声明允许在定义之前引用类型。问题在于你的代码:

class TourTable;

class Tour : public QMainWindow
{
    // ...

    TourTable _table;

};

您不仅要引用类型TourTable,还要使用TourTable _table;对其进行实例化。这需要TourTable的完整定义。

解决方案可能是在TourTable之前定义Tour,如下所示:

class TourTable : public QTableView
{
    // ...

    Tour _parent;
}

class Tour : public QMainWindow
{
    // ...

    TourTable _table;
};

但是,由于您在Tour中实例化TourTable,这只会解决问题。

根据完整的设计,解决方案可能是使用指针。这样的事情:

class TourTable;

class Tour : public QMainWindow
{
    Tour();
        // forward declaration of the constructor,
        // see below for the definition

    TourTable* _table;
        // no complete definition of TourTable at this stage,
        // only a forward declaration:
        // we can only declare a pointer
};

class TourTable : public QTableView
{
    TourTable(Tour* parent):
        QTableView(parent),
        _parent(parent)
    {
    }

    Tour* _parent;
}

Tour::Tour() // definition of Tour's constructor
{
    _table = new TourTable(this);
        // TourTable has been defined : we can instantiate it here
}