Ответ 1
Для Facility* f = new Facility();
вам нужно полное объявление, а не просто декларация.
Я знаю круговые зависимости, но даже с форвардными объявлениями я получаю эту область. Что я делаю неправильно?
// facility.h
class Area;
class Facility {
public:
Facility();
Area* getAreaThisIn();
void setAreaThisIsIn(Area* area);
private:
Area* __area;
};
// facility.cpp
#include "facility.h"
#include "area.h"
{ ... }
// area.h
class Facility;
class Area {
public:
Area(int ID);
int getId();
private:
std::list<Facility*> _facilities;
};
// area.cpp
#include "area.h"
#include "facility.h"
Итак, это компилируется отлично, но если я делаю
// foo.h
#include "facility.h"
class Foo { .. };
// foo.cpp
#include "foo.h"
void Foo::function() {
Facility* f = new Facility();
int id = f->getAreaThisIsIn()->getId();
Когда я получаю invalid use of incomplete type struct Area
Для Facility* f = new Facility();
вам нужно полное объявление, а не просто декларация.
Чтобы уточнить: переднее объявление позволяет вам работать с объектом, если оно ограничено:
struct Foo; // forward declaration
int bar(Foo* f); // allowed, makes sense in a header file
Foo* baz(); // allowed
Foo* f = new Foo(); // not allowed, as the compiler doesn't
// know how big a Foo object is
// and therefore can't allocate that much
// memory and return a pointer to it
f->quux(); // also not allowed, as the compiler doesn't know
// what members Foo has
В некоторых случаях могут помочь передовые декларации. Например, если функции в заголовке только когда-либо принимают указатели на объекты вместо объектов, тогда вам не нужно #include
определение всего класса для этого заголовка. Это может улучшить время компиляции. Но для реализации этого заголовка почти гарантированно нужно #include
соответствующее определение, потому что вы, скорее всего, захотите выделить эти объекты, вызвать методы для этих объектов и т.д., И вам нужно больше, чем прямое объявление для этого.
Вы # включали and area.h и объект .h в foo.cpp(если это файл, в котором вы получили ошибку)?