作者:兰州老头_337 | 来源:互联网 | 2023-10-15 21:51
1问题提出最近在编写“斗地主”游戏,在程序中有一个“CDealer”类用来实现发牌等等功能,还有一个“CPoker”类,用来表示扑克牌。在CDealer类中包含了一个CPoker类指针的成员变量,而
1 问题提出
最近在编写“斗地主”游戏,在程序中有一个“CDealer”类用来实现发牌等等功能,还有一个“CPoker”类,用来表示扑克牌。在CDealer类中包含了一个CPoker类指针的成员变量,而在CPoker类中也包含了一个CDealer类指针的成员变量,如下所示:
//CPoker类头文件
#pragmaonce
#include"Dealer.h"
class CPoker
{
public:
CPoker(void);
~CPoker(void);
private:
CDealer *m_pDealer;
};
//CDealer类的头文件
#pragmaonce
#include"Poker.h"
class CDealer
{
public:
CDealer(void);
~CDealer(void);
private:
CPoker *m_pPoker;
};
程序编译时,错误提示为
poker.h(10) :error C2143: 语法错误: 缺少“;”(在“*”的前面)
poker.h(10) :error C4430: 缺少类型说明符- 假定为int。注意: C++ 不支持默认int
2 问题分析
因为在程序中首先包含的是Dealer.h,所以Dealer.h头文件首先被编译。如图3-1所示。在CDealer类的头文件中,第1步执行了#pragma once之后就进入到第2步,即Poker.h头文件中。在Poker.h头文件中进行第3步,执行#pragma once,之后进入到第4步,也就是Dealer.h中,由于Dealer.h中包含#pragma once,并且此时已经被编译,所以第4步中的#include “Dealer.h”实际上没有执行,继续进行到Poker.h的第5步,定义CDealer类的指针,此时CDealer类实际上还没有定义,因此编译时回报错。
3 问题解决
3.1 类的前向声明
从2中分析可以看到,报错的主要原因是CDealer类没有定义。可以使用类的前向声明(forwarddeclaration)的方式解决这个问题。如
class CDealer;
图3-1 头文件执行流程
在CPoker类定义中映入了类类型CDealer。在类声明之后而定义之前,类CDealer是一个不完全类型(incompetetype),即已知CDealer是一个类,但不知道包含哪些成员,所以使用类的前向定义后,只能声明该类的指针,而不是该类的对象。
同理,在Dealer.h中加入CPoker类的前向声明。
classCPoker;
3.2 将包含头文件的代码放到执行文件
尽量将包含头文件的语句放到执行文件(.cpp文件)中而不是在头文件中(.h文件)。因此,接下来将#include“Poker.h”语句放到Dealer.cpp文件中,而将#include“Dealer.h”语句放到Poker.cpp文件中。
4 小结
对于两个类间相互引用引发的编译错误,可以使用类前向声明的方式解决。类前向声明是一个不完整类型,只能使用其指针。尽量将包含头文件的代码放在cpp执行文件中,而不是h头文件中。