前面我们学习了如何定义类,以及如何使用类的成员,在前面我们把成员函数都定义在了类的内部,接下来我们学习,如何把类的定义和成员函数的实现分离。
在面向对象的程序开发中,一般做法是将类的声明(其中包含成员函数的声明)放在指定的头文件中,用户如果想用该类,只要把有关的头文件包含进来即可,不必在程序中重复书写类的声明,以减少工作量,节省篇幅,提高编程的效率。
由于在头文件中包含了类的声明,因此在程序中就可以用该类来定义对象。由于在类体中包含了对成员函数的声明,在程序中就可以调用这些对象的公用成员函数。为了实现上一节所叙述的信息隐蔽,对类成员函数的定义一般不放在头文件中,而另外放在一个文件中。例如,可以分别写两个文件:
Student.h 这是头文件,在此文仅重进行类的声明
#pragma once
#include
#include
#include
#include
using namespace std;
class Student
{
private:
int id;
int age;
char name[10];
public:
void init(int sid = 0, int sage = 0, const char* sname = "NULL");
void showStuInfo();
};
Student.cpp 这是源文件,类的所有函数都在此文件中实现
#include "Student.h"
void Student::init(int sid, int sage, const char* sname)
{
id = sid;
age = sage;
strcpy(name, sname);
}
void Student::showStuInfo()
{
cout << id << " " << age << " " << name << endl;
}
为了组成一个完整的源程序,还应当有包括主函数的源文件:
#include"Student.h"//讲类声明头文件包含进来
int main()
{
Student cPlus;//定义对象
cPlus.init(111,20,"C语言Plus");
cPlus.showStuInfo();
return 0;
}
类的申明和成员函数定义分离注意问题
如果一个Student.h同时被多个文件调用,每次都要对包含成员函数定义的源文件(如上面的student.cpp)进行编译,这是否可以改进呢?的确,可以不必每次都对它重复进行编译,而只需编译一次即可。把第一次编译后所形成的目标文件保存起来,以后在需要时把它调出来直接与程序的目标文件相连接即可。这和使用函数库中的函数是类似的。这也是把成员函数的定义不放在头文件中的一个好处。(静态库,以后会讲)
在实际工作中,并不是将一个类声明做成一个头文件,而是将若干个常用的功能相近的类声明集中在一起,形成类库。类库有两种:
在程序开发工作中,类库是很有用的,它可以减少用户自己对类和成员函数进行定义的工作量。
类库包括两个组成部分:
用户只需把类库装入到自己的计算机系统中(一般装到C++编译系统所在的子目录下),并在程序中用#include命令行将有关的类声明的头文件包含到程序中,就可以使用这些类和其中的成员函数,顺利地运行程序。这和在程序中使用C++系统提供的标准函数的方法是一样的,例如用户在调用sin函数时只需将包含声明此函数的头文件包含到程序中,即可调用该库函数,而不必了解sin函数是怎么实现的(函数值是怎样计算出来的)。
当然,前提是系统已装了标准函数库。在用户源文件经过编译后,与系统库(是目标文件)相连接。在用户程序中包含类声明头文件,类声明头文件就成为用户使用类的公用接口,在头文件的类体中还提供了成员函数的函数原型声明,用户只有通过头文件才能使用有关的类。用户看得见和接触到的是这个头文件,任何要使用这个类的用户只需包含这个头文件即可。包含成员函数定义的文件就是类的实现。
请特别注意:类声明和函数定义一般是分别放在两个文本中的。由于要求接口与实现分离,为软件开发商向用户提供类库创造了很好的条件。
开发商把用户所需的各种类的声明按类放在不同的头文件中,同时对包含成员函数定义的源文件进行编译,得到成员函数定义的目标代码。软件商向用户提供这些头文件和类的实现的目标代码(不提供函数定义的源代码)。用户在使用类库中的类时,只需将有关头文件包含到自己的程序中,并且在编译后连接成员函数定义的目标代码即可。
由于类库的出现,用户可以像使用零件一样方便地使用在实践中积累的通用的或专用的类,这就大大减少了程序设计的工作量,有效地提高了工作效率。
全部0条评论
快来发表一下你的评论吧 !