1、说明:对象A对对象B进行引用,同时对象B也引用到了对象A,导致循环引用。
1.1、错误示例如下:
A.h
#pragma once
#include "B.h"
class A
{
public:
B b;
};
B.h
#pragma once
class A;
class B
{
public:
A a;
};
1.2、编译报错:
1.3、错误原因:
在A.h:2,处理语句#include “B.h”,进行头文件展开 在B.h:5进行的是在class B中声明一个A类型的成员变量,而此时class A还没有被声明 因此编译报错:‘A’ does not name a type
1.4、解决循环依赖的问题有两种方式:
1.使用前向声明(forward declaration) 2.设计层面避免循环引用
2、前项声明
2.1、前向声明的作用:
1.不需要include头文件,大量引入的头文件会导致编译变慢 2.可以解决两个类相互循环调用的情况
2.2、注意事项:
由于前向声明而没有定义的类是不完整的,所以class A只能用于定义指针、引用、或者用于函数形参的指针和引用,不能用来定义对象,或访问类的成员。 这是因为需要确定class B空间占用的大小,而类型A还没有定义不能确定大小,但A是指针类型大小已知,因此Class B中可以使用A定义成员变量。
2.3、针对前向声明的特点,修改如下:
1、将对象改成对象指针 2、B的文件中,使用A之前进行声明
示例如下: A.h
#pragma once
#include "B.h"
class A
{
public:
B* b;
};
B.h
#pragma once
class A;
class B
{
public:
A* a;
};
3、设计层面避免循环引用
说明:采用C++的多态特性,提取抽象类。使用父类指针、子类对象的方式解决循环依赖,让具体依赖抽象的父类。
实现方式如下:
IA.h
class IA
{
}
A.h
#include "IA.h"
#include "B.h"
class A : public IA
{
B* b;
}
B.h
#include "IA.h"
class B
{
public:
B(IA* ia) { m_ia = ia; }
private:
IA* m_ia;
}
main.cpp
IA* ia = new A();
|