英文原文地址

在开发大型的软件项目时,头文件需要得到恰当的管理,甚至在c中也会面临这种问题,当我们用c++开发时,头文件的管理会变得更复杂,更加耗费我们的时间去管理,下面我将讲一些包含规则来简化这个苦差事。

头文件包含规则

  • 当在在当前正在编写的文件中没有预先相应的声明时我们需要include我们需要的头文件,否则我们需要声明。
  • 头文件应该设计的与include的顺序无关.
    • 例如在写x.cpp时尽量确保x.h是x.cpp需要的第一个header file.
  • header file的包含机制允许包含重复的头文件.

接下来我们用一下例子来说明这些规则:

包含头文件范例:

下面的例子阐明了几种不同类型的依赖关系,假设我们有一个类A声明并定义在 a.ha.cpp中.

a.h


#ifndef _a_h_included_
#define _a__h_included_
#include "ABase.h"
#include "b.h"
//前置声明需要的类
class C;
class D; class A : public ABase
{
B m_b;
C *m_c;
D *m_d;
public:
void SetC(C *c);
C *GetC() const; void ModifyD(D*d);
}
#endif

a.cpp

    #include "a.h"
#include "d.h" void A :: SetC(C*c)
{
m_c = c;
} C *A :: GetC() const
{
return m_c;
} void A :: ModifyD(D *d)
{
d -> SetX(0);
d ->SetY(0);
m_d = d;
}

对头文件包含过程的分析

**class Bbase** ABase是基类,需要被A继承,因此Abase需要完整的声明,编译器需要知道ABase对象的大小才能确定类A对象的大小,所以在a.h中应当显式地包含ABase.h **class B** 类B的对象是类A的数据成员,因此类B也需要完整的声明,因为编译器在确定类A对象大小时需要知道类B对象的大小,因此类B也需要显式地包含 **class C** 类c在类A中是以指针类型出现的,在这里a.h或者a.cpp对类c内容的大小不关心,这时候一个前置的类c声明放在a.h中就可以了,在上面a.h和a.cpp中也并没有显式的包含c.h **class D** 类D和类c一样也是作为指针在a.h中出现,所以也只需要在a.h中做前置的声明,但是在a.cpp中用了类D的对象,所以在a.cpp中需要显示的包含d.h

关键点

  • 当需要类的实例参与程序的设计时我们需要显示的包含相应的头文件,但如果是指针我们可以使用前置声明。
  • 在a.cpp中a.h是第一个被包含的头文件,a.h不希望有其他的头文件包含在它前面,由于a.h在a.cpp中作为了第一个包含的头文件,为了程序的正确编译,我们不允许有其他任何的头文件在a.h以前被包含。
    • 如果在所有的类中都遵循这个规则(换句话说就是x.cpp总是将x.h作为第一个包含的头文件),那么将不会产生各个头文件包含的依赖关系。
  • 在a.h中的预处理过程(a_h_include),允许头文件的重复包含。

循环依赖

下面的class X 和 class Y存在循环依赖, 我们通过前置声明来解决这个问题

    /*=======x.h===========*/
class Y; class X
{
Y *m_y;
.....
};
/*=====y.h============*/
class X;
class Y
{
X *m_x;
.......
};

c++中头文件include规则浅析[译]的更多相关文章

  1. Visual Studio 中使用万能头文件 #include <bits/stdc++.h>

    最近开始使用VS,之前用的DEV C++软件可直接使用 #include <bits/stdc++.h>  ,但VS中并没有,为了使用方便,可直接在VS中添加此头文件,方法如下: 1.在安 ...

  2. C++中头文件与源文件的作用详解

    一.C++ 编译模式 通常,在一个 C++ 程序中,只包含两类文件―― .cpp 文件和 .h 文件.其中,.cpp 文件被称作 C++ 源文件,里面放的都是 C++ 的源代码:而 .h 文件则被称作 ...

  3. C语言中头文件怎么写?(本文来源网络,由黑乌鸦进一步完善)

      c语言头文件怎么写?我一直有这样的疑问,但是也一直没去问问到底咋回事:所以今天一定要把它弄明白! 其实学会写头文件之后可以为我们省去不少事情,可以避免书写大量的重复代码.有利于整理思路.使代码脉络 ...

  4. C/C++ 中头文件相互包含引发的问题

    转自:http://blog.csdn.net/hazir/article/details/38600419 今天下午遇到一个头文件相互包含而导致的编译问题,花了我不少时间去调试没找到问题,最后晚上跟 ...

  5. C++ 中头文件(.h)和源文件(.cc)的写法简述

    用C++编写比较大型的项目时,文件的分割管理确实确实是非常必要的 .下面就非常简洁明了地谈谈头文件(.h)和源文件(.cc)应该怎么写. 头文件(.h):写类的声明(包括类里面的成员和方法的声明).函 ...

  6. 万能头文件#include<bits/stdc++.h>

    最近在打cf时赛后翻阅别人的代码总是会发现一个陌生而奇怪的头文件#include<bits/stdc++.h> 奇怪之处就在于基本上所有的代码只要用了这个头文件就不再写其他头文件了. 百度 ...

  7. c中头文件在cpp文件里引用和.h文件引用的思考

    我们在编敲代码中头文件是常常使用的. 可是头文件是应该包括在.H文件里还是在.cpp文件里.在这个其中有什么样去差别呢. 假如说我们编写了一个a.cpp  .我们将a.cpp文件的变量和函数申明在a. ...

  8. C语言中头文件<stdio.h>中的#ifndef _STDIO_H_

    先了解这里的相关知识:http://www.cnblogs.com/stemon/p/4000468.html 头文件的中的#ifndef,这是一个很关键的东西.比如你有两个C文件,这两个C文件都in ...

  9. C语言中头文件和cpp文件解析

    务必提前预读这里的内容:http://www.cnblogs.com/stemon/p/3999844.html 回到cpp文件与头文件各写什么内容的话题上: 理论上来说cpp文件与头文件里的内容,只 ...

随机推荐

  1. github的入门使用

    原文 http://www.eoeandroid.com/thread-274556-1-1.html [初识Github]首先让我们大家一起喊一句“Hello Github”.YEAH!就是这样. ...

  2. hdu1505

    the main algorithm as the 1506 #include <stdio.h> #include <iostream> #include <strin ...

  3. Android IOS WebRTC 音视频开发总结(四十)-- 国内webrtc现状

    本文主要介绍目前国内webrtc开发现状,文章来自博客园rtc.blacker,支持原创,转载必须说明出处. 上次一国外合作伙伴问我国内rtc应用和开发状况怎样,哪些城市比较火,那些行业应用比较多,我 ...

  4. 二、搭建struts2的开发环境

    二.搭建struts2的开发环境 下载地址:http://struts.apache.org 解压后的目录结构: apps:框架本身提供一些案例(学习) docs:框架本身提供的文档(指南和API). ...

  5. 实现弹出收回菜单效果ios源码

    REMenu能够提供下弹出来的菜单,跳转到不同的vc后菜单便会收起.菜单的弹收都有回弹(bounce)的效果.效果图: <ignore_js_op> 使用方法: 先把REMenu的文件夹复 ...

  6. Cadence仿真利器,Cadence SI / PI Analysis – Sigrity安装及破解指南

    Sigrity提供了丰富的千兆比特信号与电源网络分析技术,包括面向系统.印刷电路板(PCB)和IC封装设计的独特的考虑电源影响的信号完整性分析功能. Sigrity分析技术与Cadence Alleg ...

  7. 查看Linux系统版本信息

    一.查看Linux内核版本命令(两种方法): 1.cat /proc/version [root@S-CentOS home]# cat /proc/versionLinux version 2.6. ...

  8. VS2008无法切换到视图设计器

    编写人:CC阿爸 2014-2-17 近来用于干活的笔记本电脑实在太慢了,在领导的安排下,有幸更换了一台配置好的电脑.经过一天的努力,所有之前的开发软件都安装完成了.并且OS从xp升级到win7.SQ ...

  9. table中绝对定位元素相对td定位失效解决方案

    开门见山! 问题:在一个table中,我需要在td里面绝对定位一个div, 写法:td{position:relative;} div{position:absolute;} OK,就这么简单,思路也 ...

  10. scala学习笔记2

    一.算术和操作符重载 a + b 是如下方法的简写: a.+(b) 在scala中你可以使用任何符号来为方法命名.比如BigInt类就定义了一个/%的方法,该方法返回一个对偶,对偶的内容是除法操作得到 ...