C++前置声明

【1】一般的前置函数声明

见过最多的前置函数声明,基本格式代码如下:

 1 #include <iostream>
 2 using namespace std;
 3
 4 void fun(char ch, int *pValue, double dValue);
 5
 6 void main()
 7 {
 8     int nValue = 100;
 9     double dValue = 111.22;
10     fun(‘a‘, &nValue, dValue);
11
12     system("pause");
13 }
14
15 void fun(char ch, int *pValue, double dValue)
16 {
17     return;
18 }

很好理解,不做赘述。

【2】自定义类型的前置声明

自定义类型的前置声明,由于编译器不知道类型的大小,所以不可以声明类型的对象。只可以利用类型声明指针和引用。

代码如下:

 1 /*
 2  * 自定义类型前置声明
 3  */
 4 #include <iostream>
 5 using namespace std;
 6
 7 class B;
 8
 9 class A
10 {
11 private:
12     // 内置类型
13     int m_nInt;
14     int& m_nRInt;
15     int* m_pInt;
16
17     // 自定义类型
18 //    B b; // error!
19     B* m_pB;
20     B& m_b;
21
22 public:
23     A (B *pBPara = NULL) : m_nInt(100)
24         , m_nRInt(m_nInt)
25         , m_pInt(NULL)
26         , m_pB(NULL)
27         , m_b((NULL == pBPara) ? (*m_pB) : (*pBPara))
28     {
29         cout << "A()" << endl;
30     }
31     ~A()
32     {
33         cout << "~A()" << endl;
34     }
35
36     void funA()
37     {
38 //        m_pB->doAnything(); // build error C2027: use of undefined type ‘B‘
39     }
40 };
41
42 class B
43 {
44 private:
45     int m_n;
46
47 public:
48     B (int n = 100) : m_n(n)
49     {
50         cout << "B()" << endl;
51     }
52     ~B()
53     {
54         cout << "~B()" << endl;
55     }
56     void doAnything()
57     {
58         cout << "B::anythig()" << endl;
59     }
60 };
61
62 void main()
63 {
64     A objA;
65     system("pause");
66 }

如上,利用前置类型的指针想调用其成员函数,会报编译错误!那么,肿么办?请看下文。

【3】声明和实现分离

代码如下,声明头文件:

 1 /*
 2  * TestForwardDeclar.h
 3  */
 4 #ifndef D_TESTFORWARDDECLAR_H_
 5 #define D_TESTFORWARDDECLAR_H_
 6
 7 #include <iostream>
 8
 9 class B;  // 前置声明自定义类型
10
11 class A
12 {
13 private:
14     // 内置类型
15     int m_nInt;
16     int& m_nRInt;
17     int* m_pInt;
18
19     // 自定义类型
20 //    B b; // error!
21     B* m_pB;
22     B& m_b;
23
24 public:
25     A (B *pBPara = NULL);
26     ~A ();
27     void funA();
28 };
29
30 class B
31 {
32 private:
33     int m_n;
34
35 public:
36     B (int n = 100);
37     ~B ();
38     void doAnything();
39 };
40
41 #endif

代码如下,定义文件:

 1 /*
 2  * TestForwardDeclar.cpp
 3  */
 4
 5 #include "TestForwardDeclar.h"
 6 #include <iostream>
 7
 8 A::A (B *pBPara)
 9     : m_nInt(100)
10     , m_nRInt(m_nInt)
11     , m_pInt(NULL)
12     , m_pB(NULL)
13     , m_b((NULL == pBPara) ? (*m_pB) : (*pBPara))
14 {
15     std::cout << "A()" << std::endl;
16 }
17
18 A::~A()
19 {
20     std::cout << "~A()" << std::endl;
21 }
22
23 void A::funA()
24 {
25     m_pB->doAnything();  // 分开头文件和实现文件即可
26 }
27
28
29 B::B (int n) : m_n(n)
30 {
31     std::cout << "B()" << std::endl;
32 }
33
34 B::~B()
35 {
36     std::cout << "~B()" << std::endl;
37 }
38
39 void B::doAnything()
40 {
41     std::cout << "B::anythig()" << std::endl;
42 }

代码如下:测试文件:

1 #include "TestForwardDeclar.h"
2
3 void main()
4 {
5     A objA;
6 }

编译成功,运行结果是期望效果。

【4】总结

自定义类型前置声明时,只可以利用类型名声明指针和引用变量(谨记不可以声明对象或new 对象,均因为类型大小不确定,编译器无能为力)。

若需要利用指针或引用调用前置类型的接口,必须按照声明和实现分离的方式进行编码。

Good Good Study, Day Day Up.

顺序 选择 循环 总结

时间: 2024-08-04 09:08:30

C++前置声明的相关文章

c++ 类前置声明【转】

[转自 here] 在编写C++程序的时候,偶尔需要用到前置声明(Forward declaration).下面的程序中,带注释的那行就是类B的前置说明.这是必须的,因为类A中用到了类B,而类B的声明出现在类A的后面.如果没有类B的前置说明,下面的程序将不同通过编译,编译器将会给出类似"缺少类型说明符"这样的出错提示. 代码一: // ForwardDeclaration.h #include <iostream> using namespace std; class B;

C++中类的前置声明和包含头文件的区别

一.类嵌套的疑问 C++头文件重复包含实在是一个令人头痛的问题,假设我们有两个类A和B,分别定义在各自的头文件A.h和B.h中,但是在A中要用到B,B中也要用到A,但是这样的写法当然是错误的: class B; class A{ public: B b;}; class B{ public: A a;}; 因为在A对象中要开辟一块属于B的空间,而B中又有A的空间,是一个逻辑错误,无法实现的,在这里我们只需要把其中的一个A类中的B类型成员改成指针形式就可以避免这个无限延伸的怪圈了,为什么要更改A而

头文件前置声明错误解决

  如上图,np_utility.h中使用了QString的前置声明,np_utility.cpp文件中包含了QString的头文件,本以为这样没问题,编辑就报错了,后来发现,QString头文件放在np_utility.cpp文件中的np_utility.h之前,QString的前置声明才不会报错. 原因是: 本来我的前置声明就是在np_utility.h这个文件里声明的,只要qstring声明在np_utility.h之前就可以了.

C++ 类声明 类前置声明范例

转载自http://www.cnblogs.com/staring-hxs/p/3244251.html 在编写C++程序的时候,偶尔需要用到前置声明(Forward declaration).下面的程序中,带注释的那行就是类B的前置说明.这是必须的,因为类A中用到了类B,而类B的声明出现在类A的后面.如果没有类B的前置说明,下面的程序将不同通过编译,编译器将会给出类似"缺少类型说明符"这样的出错提示. 代码一: // ForwardDeclaration.h #include <

前置声明和头文件

假设有一个Date类 Date.h class Date {   private:       int year, month, day;   };   如果有个Task类的定义要用到Date类,有两种写法 其一 Task1.h class Date;   class Task1 {   public:       Date getData();   };   其二 Task2.h #include "Date.h"   class Task2 {   public:       Da

C++_前置声明

为什么要有前置声明? eg: -定义一个类 class A,这个类里面使用了类B的对象b,然后定义了一个类B,里面也包含了一个类A的对象a,就成了这样: //a.h #include "b.h" class A { .... private: B b; }; //b.h #include "a.h" class B { .... private: A a; }; 一编译,就出现了一个相互包含的问题了,解决的方法是在a.h文件中声明类B,然后使用B的指针. //a.h

QDemo之前置声明

前置声明 = Forward Declaration 对于一个刚刚接触include写法的童鞋来说, 突然看到很多人用如下写法:不知甚解否? #ifndef DIALOG_H #define DIALOG_H #include <QDialog> //! [前置声明] QT_BEGIN_NAMESPACE class QPushButton; class QLabel; class QLineEdit; QT_END_NAMESPACE //! [前置声明] class Dialog:publ

模板链接与前置声明引发的血案

模板链接与前置声明引发的血案 模板链接与前置声明引发的血案 现象 问题原型 模板參数类型类 使用类模板的类 分析 objdump -S TemplateLink SUPERSUBCLASS 分析 objdump -S UsingBaseo objdump -S UsingChildo 问题解答 解答问题一 解答问题二 解决方式 类型萃取辅助类 应用 不足 现象: 有一个类模板,它会依据模板类型參数T的实际类型,调用不同的实例化泛型函数子去处理实际事情. 在程序运行时.发如今不同的模块中用相同的类

Google C++编程规范 – 第十九条 -《前置声明》

转自:http://roclinux.cn/?p=3285 本原创文章属于<Linux大棚>博客. 博客地址为http://roclinux.cn. 文章作者为roc wu == [规范] 对于普通的类,建议使用前置声明,而不是#include. [什么是前置声明] 在英文中,前置声明称为“forward declaration”,是指“对类.函数或模板进行声明,且不含相关的具体定义”.我们可以使用前置声明来代替那些用于声明的#include语句. [支持者的声音] 过多的#include会导