浅谈头文件(.h)和源文件(.cpp)的区别

浅谈头文件(.h)和源文件(.cpp)的区别

本人原来在大一写C的时候,都是所有代码写在一个文件里一锅乱煮。经过自己开始写程序之后,发现一个工程只有一定是由多个不同功能、分门别类展开的文件构成的。一锅乱煮只会导致代码可读性差、维护性差。但是本人在学习C++的时候,被这章节弄混了。 Source & Header, Separate Files from Classes,里面介绍了一个对于Python写手来说很新的概念。就是头文件和源文件的区别,下面谈谈我自己学习后的理解。

头文件(.h)

头文件用来写 类的声明 (包括类的成员的声明和方法声明)、函数原型#define 常数等,但是很少会写出具体的实现和细节。就好比抽象类一样。

头文件很有意思的是,开头和结尾必须按照以下格式:

#ifndef MYCLASS_H
#define MYCLASS_H

// code here

#endif

当时我看到这个是极其的不理解和迷茫的,后来阅读了别人的博文才略懂。

首先解释他是干嘛使的,这是防止头文件被重复引用。什么叫被重复引用?就是同一个头文件(.h)在同一个源文件(.cpp)中被include了多次。这种错误常常是因为include嵌套。举个最简单的例子,存在cellphone.h这个头文件引用了#include "huawei.h",之后又有china.cpp这个源文件同时导入了#include "cellphone.h" 和 #include "huawei.h"。此时huawei.h就在一个源文件里引用了两次。

那么,某些时候,只是因为include了两遍,增大了编译器的工作量。如果是小型程序的话还好说,但是大型工程甚至会增长几个小时的编译时间。但是另一些情况,会引起很严重的错误。比如在头文件中定义了全局变量会引起重复定义。

所以就有了我们上面那些看起来乱七八糟的代码,下面开始解释。

#ifndef MYCLASS_H 的意思是 if not define myclass.h,这样看就很好理解了,如果引用这个头文件的源文件不存在myclass.h这个头文件,那么接下行 #define MYCALSS_H, 引入myclass.h。然后就是我们头文件的代码。如果已经有了,直接跳到 #endif。

理论上来说,上面这个片段的MYCLASS_H是可以任意命名的,但是约定俗成的,为了可读性的,我们都把它命名为这个头文件的大写和下划线的形式。如下面这一段代码:

#ifndef HUAWEI_H       // 防止huawei.h被重复引用
#define HUAWEI_H

#include <cmath>       // 引用标准库
#include "honor.h"     // 引用非标准库头文件
...
void Function();  	   // 全局函数声明
class Mate20{		   // 类声明
    public: Mate20();  // 构造函数声明
 			~Mate20(); // 析构函数声明  
    private:
    protected:
};

#endif

上面的代码其实已经很好的解释了头文件的作用——声明。可以看见内部的函数和方法仅仅是声明,而都没有写入具体细节。

源文件(.cpp)

源文件主要写实现头文件中已经声明的那些函数的具体代码。需要注意的是,开头必须#include一下实现的头文件,以及要用到的头文件。那么当你需要用到自己写的头文件中的类时,只需要#include进来就行了。

下面举个例子

我首先写一个头文件Pen.h,里面定义了一个Pen类:

#ifndef PEN_H
#define PEN_H
#include <string>
class Pen
{
public:
	Pen(std::string brand);

};

#endif

然后我在我的主文件test.cpp中引入了这个头文件:

#include <iostream>
#include "Pen.h"
using namespace std;

Pen::Pen(string brand) {
	cout << "What brand are you looking for?" << endl;
	cout << brand;
}	

int main() {
	Pen pilot("pilot");
	
}

// 输出结果为 
/* What brand are you looking for?
   pilot */

结果表明,这样引入是正确的。

posted @ 2020-02-09 14:45  scyq  阅读(19643)  评论(0编辑  收藏  举报