C++中前置声明和头文件包含

转载 2015年12月21日 20:40:24

假设有一个Date类

Date.h

  1. class Date {  
  2. private:  
  3.     int year, month, day;  
  4. };  

如果有个Task类的定义要用到Date类,有两种写法

其一

Task1.h

  1. class Date;  
  2. class Task1 {  
  3. public:  
  4.     Date getData();  
  5. };  

其二

Task2.h

  1. #include "Date.h"  
  2. class Task2 {  
  3. public:  
  4.     Date getData();  
  5. };  

一个采用前置声明,一个采用#include<Date.h>加入了Date的定义。两种方法都能通过编译。但是 Task1.h 这种写法更好。如果Date.h 的 private 成员变量改变,比如变成 double year, month, day; ,Task1.h 不需要重新编译,而 Task2.h 就要重新编译,更糟的是如果 Task2.h 还与其他很多头文件有依赖关系,就会引发一连串的重新编译,花费极大的时间。可是事实上改变一下写法就可以省去很多功夫。所以能用前置声明代替#include 的时候,尽量用前置声明


有些情况不能用前置声明代替#include

比如Task1.h改成

  1. class Date;  
  2. class Task1 {  
  3. public:  
  4.     Date d;  
  5. };  

会编译错误,因为Date d定义了一个Date类型变量,编译器为d分配内存空间的时候必须知道d的大小,必须包含定义Date类的Date.h文件

这是可以采用指针来代替

  1. class Date;  
  2. class Task1 {  
  3. public:  
  4.     Date *d;  
  5. };  

指针的大小是固定的。在32位机上是4字节,64位机上是8字节。这时编译Task1的时候不需要Date的大小,所以和Date的定义无关。

上述例子可以说明

如果使用object reference 或 object point 可以完成任务,就不要用object

这样可以尽最大可能避免#include

为声明式和定义是提供不同的头文件 

在函数库的设计过程中,接口的设计就要遵循上述准则。

一个接口的头文件是这样的

interface.h

  1. class Date;  
  2. class Address;  
  3. class Email;  
  4. Date getDate();  

如果客户只用到Date类,编译器就只会去编译Date.h,而不去编译Address.h,Email.h 等等文件。


究竟什么时候需要前置声明,什么时候需要头文件包含呢?

      头文件包含其实是一想很烦琐的工作,不但我们看着累,编译器编译的时候也很累,再加上头文件中常常出现的宏定义。感觉各种宏定义的展开是非常耗时间的,远不如自定义函数来得速度。我仅就不同头文件、源文件间的句则结构问题提出两点原则,仅供参考:

  第一个原则应该是,如果可以不包含头文件,那就不要包含了。这时候前置声明可以解决问题。如果使用的仅仅是一个类的指针,没有使用这个类的具体对象(非指针),也没有访问到类的具体成员,那么前置声明就可以了。因为指针这一数据类型的大小是特定的,编译器可以获知。

  第二个原则应该是,尽量在CPP文件中包含头文件,而非在头文件中假设类A的一个成员是是一个指向类B的指针,在类A的头文件中使用了类B的前置声明并编译成功,那么在A的实现中我们需要访问B的具体成员,因此需要包含头文件,那么我们应该在类A的实现部分(CPP文件)包含类B的头文件而非在声明部分(H文件)包含。


如何使用前置声明取代包括头文件

这篇文章很大程度是受到Exceptional C++ (Hurb99)书中第四章 Compiler  Firewalls and the Pimpl Idiom  (编译器防火墙和Pimpl惯用法) ...
  • rogeryi
  • rogeryi
  • 2006年12月12日 12:09
  • 9172

QT 5.3 应用程序打包成可执行程序exe

众所周知QT编译好的应用程序依赖很多动态链接库,不能方便的在其他PC上使用。这里介绍QT5版本的应用程序打包成可执行EXE的方法。 首先把QT应用程序项目编译成Release版本,而不是Debug版...

【机器学习】神经网络(二)——反向传播算法

反向传播算法是目前用来训练人工神经网络的最常用且最有效的算法。

[C++]前置声明和头文件

假设有一个Date类 Date.h [cpp] view plain copy class Date {   private:      ...

[C++]前置声明和头文件

看完《Effective C++》条款31有感。。假设有一个Date类Date.hclass Date { private: int year, month, day; };如果有个Task类...

C++类中使用前置声明和使用include包含头文件的感想

C++前置声明和include包含辨析
  • jh1513
  • jh1513
  • 2016年09月26日 21:21
  • 199

前置声明与C++头文件互相包含导致的error: 'xxx' does not name a type问题

在一个源文件中,要声明或定义一个类的指针时,必须在使用前声明或定义该类,因此下面的代码会报错:class A { public: B *b; };class B { public: A...
  • imred
  • imred
  • 2017年01月17日 15:39
  • 3198

关于前置声明与C++中头文件相互包含的几点问题

一个前置声明是指在程序员尚未给出完整定义之前对一个标示符(一个类型、一个变量或者一个函数)的声明。一个很简单的例子就是我们在函数A中使用了函数B,但是函数B的声明在函数A之后,这个时候,就需要对函数B...

C++中头文件相互包含与前置声明

一、类嵌套的疑问 C++头文件重复包含实在是一个令人头痛的问题,前一段时间在做一个简单的数据结构演示程序的时候,不只一次的遇到这种问题。假设我们有两个类A和B,分别定义在各自的有文件A.h和B.h中,...
  • wjs1033
  • wjs1033
  • 2016年04月22日 12:01
  • 337

C++中头文件中是包含include还是包含类的前置声明

在一些大的工程中,可能会包含几十个基础类,免不了之间会互相引用 ( 不满足继承关系,而是组合关系 ) 。也就是需要互相声明。好了,这时候会带来一些混乱。如果处理得不好,会搞得一团糟,根据我的经验,简单...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:C++中前置声明和头文件包含
举报原因:
原因补充:

(最多只允许输入30个字)