我有2个头文件和一个源文件,
food.h:
#ifndef _FOOD_H
#define _FOOD_H
struct food {
char name[10];
};
#endif
dog.h:
#ifndef _DOG_H
#define _DOG_H
class food; //forward declaration
class Dog {
public:
Dog(food f);
private:
food _f;
};
#endif
这是class dog
的源文件,
//dog.cpp
#include "dog.h"
#include "food.h"
Dog::Dog(food f) : _f(f)
{}
问题是我可以编译上面的代码,我收到错误说_f has incomplete type
。
我以为我可以在dog.h
中提出前瞻性声明,并在food.h
中加入dog.cpp
,但这不起作用,为什么?我不应该将用户定义的头文件放在.h
文件中,对吧?它被弃用了,不是吗?
答案 0 :(得分:3)
在这种情况下,前向声明不起作用,因为您的Dog
类的实例为food
。编译器需要food
的完整声明才能声明Dog
。
关于包括弃用的问题,除非我误解了某些内容,否则根本不是这样。在此处包含food.h
可以解决您的问题:
//dog.h
#ifndef DOG_H
#define DOG_H
#include "food.h";
class Dog {
public:
Dog(food f);
private:
food f_;
};
答案 1 :(得分:2)
编译器需要知道仅使用前向声明无法访问的对象_f
的大小。如果您使用指向class food
的指针,这将有效,但由于您的成员是一个完整的类实例,您需要包含标题,以便编译器可以知道class food
的大小。
同样,如果您从class food
继承,前向声明将无效 - 您将特别需要包含标题。
只需删除前向声明和#include "food.h"
,您的代码就会编译。