1、面向对象1、什么是面向对象到目前为止,我们介绍的是 C+在面向过程的程序设计中的应用。对于规模比较小的程序,编程者可以直接编写出一个面向过程的程序,详细地描述每一瞬时的数据结构及对其的操作过程。但是当程序规模较大时,就显得力不从心了。C+就是为了解决编写大程序过程中的困难而产生的。面向对象的程序设计的思路和人们日常生活中处理问题的思路是相似的。在自然世界和社会生活中,一个复杂的事物总是由许多部分组成的。当人们生产汽车时,分别设计和制造发动机、底盘、车身和轮子,最后把它们组装在一起。在组装时,各部分之间有一定的联系,以便协调工作。这就是面向对象的程序设计的基本思路。为了进一步说明问题,下面先讨论
2、几个有关的概念。 对象客观世界中任何一个事物都可以看成一个对象(object)。对象可大可小。对象是构成系统的基本单位。任何一个对象都应当具有这两个要素,即属性(attribute)和行为(behavior),它能根据外界给的信息进行相应的操作。一个对象往往是由一组属性和一组行为构成的。一般来说,凡是具备属性和行为这两种要素的,都可以作为对象。在一个系统中的多个对象之间通过一定的渠道相互联系,如图8.1示意。要使某一个对象实现某一种行为(即操作),应当向它传送相应的消息。对象之间就是这样通过发送和接收消息互相联系的。 图8.1 面向对象的程序设计采用了以上人们所熟悉的这种思路。使用面向对象的程
3、序设计方法设计一个复杂的软件系统时,首要的问题是确定该系统是由哪些对象组成的,并且设计这些对象。在 C+中,每个对象都是由数据和函数(即操作代码)这两部分组成的,见图8.2。 图8.2 数据体现了前面提到的“属性” ,如一个三角形对象,它的3个边长就是它的属性。函数是用来对数据进行操作的,以便实现某些功能,例如可以通过边长计算出三角形的面积,并且输出三角形的边长和面积。计算三角形面积和输出有关数据就是前面提到的行为,在程序设计方法中也称为方法(method)。调用对象中的函数就是向该对象传送一个消息(message),要求该对象实现某一行为(功能)。 封装与信息隐蔽可以对一个对象进行封装处理,
4、把它的一部分属性和功能对外界屏蔽,也就是说从外界是看不到的,甚至是不可知的。这样做的好处是大大降低了操作对象的复杂程度。面向对象程序设计方法的一个重要特点就是“封装性” (encapsulation),所谓“封装” ,指两方面的含义:一是将有关的数据和操作代码封装在一个对象中,形成一个基本单位,各个对象之间相对独立,互不干扰。二是将对象中某些部分对外隐蔽,即隐蔽其内部细节,只留下少量接口,以便与外界联系,接收外界的消息。这种对外界隐蔽的做法称为信息隐蔽(imformation hiding)。信息隐蔽还有利于数据安全,防止无关的人了解和修改数据。C+的对象中的函数名就是对象的对外接口,外界可以
5、通过函数名来调用这些函数来实现某些行为(功能)。这些将在以后详细介绍。 抽象在程序设计方法中,常用到抽象(abstraction)这一名词。抽象的过程是将有关事物的共性归纳、集中的过程。抽象的作用是表示同一类事物的本质。C 和 C+中的数据类型就是对一批具体的数的抽象。对象是具体存在的,如一个三角形可以作为一个对象,10个不同尺寸的三角形是10个对象。如果这10个三角形对象有相同的属性和行为,可以将它们抽象为一种类型,称为三角形类型。在 C+中,这种类型就称为“类(class )”。这10个三角形就是属于同一“类”的对象。类是对象的抽象,而对象则是类的特例,或者说是类的具体表现形式。 继承与重
6、用如果在软件开发中已经建立了一个名为 A 的“类” ,又想另外建立一个名为 B 的“类” ,而后者与前者内容基本相同,只是在前者的基础上增加一些属性和行为,只需在类 A的基础上增加一些新内容即可。这就是面向对象程序设计中的继承机制。利用继承可以简化程序设计的步骤。“白马”继承了“马”的基本特征,又增加了新的特征(颜色), “马”是父类,或称为基类, “白马”是从“马”派生出来的,称为子类或派生类。C+提供了继承机制,采用继承的方法可以很方便地利用一个已有的类建立一个新的类。这就是常说的“软件重用”(software reusability) 的思想。 多态性如果有几个相似而不完全相同的对象,有
7、时人们要求在向它们发出同一个消息时,它们的反应各不相同,分别执行不同的操作。这种情况就是多态现象。如,在 Windows 环境下,用鼠标双击一个文件对象(这就是向对象传送一个消息),如果对象是一个可执行文件,则会执行此程序,如果对象是一个文本文件,则启动文本编辑器并打开该文件。在 C+中,所谓多态性(polymorphism)是指:由继承而产生的相关的不同的类,其对象对同一消息会作出不同的响应。多态性是面向对象程序设计的一个重要特征,能增加程序的灵活性。2、面向对象程序设计的特点传统的面向过程程序设计是围绕功能进行的,用一个函数实现一个功能。所有的数据都是公用的,一个函数可以使用任何一组数据,
8、而一组数据又能被多个函数所使用(见图 8.3) 。 图8.3面向对象程序设计采取的是另外一种思路。它面对的是一个个对象。实际上,每一组数据都是有特定的用途的,是某种操作的对象。也就是说,一组操作调用一组数据。程序设计者的任务包括两个方面: 设计所需的各种类和对象,即决定把哪些数据和操作封装在一起; 考虑怎样向有关对象发送消息,以完成所需的任务。这时他如同一个总调度,不断地向各个对象发出命令,让这些对象活动起来(或者说激活这些对象),完成自己职责范围内的工作。各个对象的操作完成了,整体任务也就完成了。显然,对一个大型任务来说,面向对象程序设计方法是十分有效的,它能大大降低程序设计人员的工作难度,
9、减少出错机会。 3、类和对象的作用类是 C+中十分重要的概念,它是实现面向对象程序设计的基础。类是所有面向对象的语言的共同特征,所有面向对象的语言都提供了这种类型。一个有一定规模的 C+程序是由许多类所构成的。C+支持面向过程的程序设计,也支持基于对象的程序设计,又支持面向对象的程序设计。以后我们将介绍基于对象的程序设计。包括类和对象的概念、类的机制和声明、类对象的定义与使用等。这是面向对象的程序设计的基础。基于对象就是基于类。与面向过程的程序不同,基于对象的程序是以类和对象为基础的,程序的操作是围绕对象进行的。在此基础上利用了继承机制和多态性,就成为面向对象的程序设计(有时不细分基于对象程序
10、设计和面向对象程序设计,而把二者合称为面向对象的程序设计)。基于对象程序设计所面对的是一个个对象。所有的数据分别属于不同的对象。在面向过程的结构化程序设计中,人们常使用这样的公式来表述程序:程序=算法数据结构算法和数据结构两者是互相独立、分开设计的,面向过程的程序设计是以算法为主体的。在实践中人们逐渐认识到算法和数据结构是互相紧密联系不可分的,应当以一个算法对应一组数据结构,而不宜提倡一个算法对应多组数据结构,以及一组数据结构对应多个算法。基于对象和面向对象程序设计就是把一个算法和一组数据结构封装在一个对象中。因此,就形成了新的观念:对象 = 算法 数据结构程序 = (对象+对象+对象+) +
11、 消息或:程序 = 对象 s + 消息“对象 s”表示多个对象。消息的作用就是对对象的控制。程序设计的关键是设计好每一个对象,及确定向这些对象发出的命令,使各对象完成相应操作。4、面向对象的软件开发随着软件规模的迅速增大,软件人员面临的问题十分复杂。需要规范整个软件开发过程,明确软件开发过程中每个阶段的任务,在保证前一个阶段工作的正确性的情况下,再进行下一阶段的工作。这就是软件工程学需要研究和解决的问题。面向对象的软件工程包括以下几个部分: 面向对象分析(object oriented analysis,OOA)软件工程中的系统分析阶段,系统分析员要和用户结合在一起,对用户的需求作出精确的分析
12、和明确的描述,从宏观的角度概括出系统应该做什么(而不是怎么做)。面向对象的分析,要按照面向对象的概念和方法,在对任务的分析中,从客观存在的事物和事物之间的关系,归纳出有关的对象(包括对象的属性和行为)以及对象之间的联系,并将具有相同属性和行为的对象用一个类(class )来表示。建立一个能反映真实工作情况的需求模型。 面向对象设计(object oriented design,OOD)根据面向对象分析阶段形成的需求模型,对每一部分分别进行具体的设计,首先是进行类的设计,类的设计可能包含多个层次(利用继承与派生)。然后以这些类为基础提出程序设计的思路和方法,包括对算法的设计。在设计阶段,并不牵涉
13、某一种具体的计算机语言,而是用一种更通用的描述工具(如伪代码或流程图)来描述。 面向对象编程(object oriented programming, OOP)根据面向对象设计的结果,用一种计算机语言把它写成程序,显然应当选用面向对象的计算机语言(例如 C+),否则无法实现面向对象设计的要求。 面向对象测试(object oriented test,OOT)在写好程序后交给用户使用前,必须对程序进行严格的测试。测试的目的是发现程序中的错误并改正它。面向对象测试是用面向对象的方法进行测试,以类作为测试的基本单元。 面向对象维护(object oriented soft maintenance,
14、OOSM)因为对象的封装性,修改一个对象对其它对象影响很小。利用面向对象的方法维护程序,大大提高了软件维护的效率。现在设计一个大的软件,是严格按照面向对象软件工程的5个阶段进行的,这5个阶段的工作不是由一个人从头到尾完成的,而是由不同的人分别完成的。这样,OOP 阶段的任务就比较简单了,程序编写者只需要根据 OOD 提出的思路用面向对象语言编写出程序即可。在一个大型软件的开发中,OOP 只是面向对象开发过程中的一个很小的部分。如果所处理的是一个较简单的问题,可以不必严格按照以上5个阶段进行,往往由程序设计者按照面向对象的方法进行程序设计,包括类的设计(或选用已有的类)和程序的设计。五、C+类的
15、声明和对象的定义一、类和对象的关系每一个实体都是对象。有一些对象是具有相同的结构和特性的。每个对象都属于一个特定的类型。在 C+中对象的类型称为类(class )。类代表了某一批对象的共性和特征。前面已说明:类是对象的抽象,而对象是类的具体实例(instance)。正如同结构体类型和结构体变量的关系一样,人们先声明一个结构体类型,然后用它去定义结构体变量。同一个结构体类型可以定义出多个不同的结构体变量。在 C+中也是先声明一个类类型,然后用它去定义若干个同类型的对象。对象就是类类型的一个变量。可以说类是对象的模板,是用来定义对象的一种抽象类型。类是抽象的,不占用内存,而对象是具体的,占用存储空
16、间。在一开始时弄清对象和类的关系是十分重要的。二、声明类类型类是用户自己指定的类型。如果程序中要用到类类型,必须自己根据需要进行声明,或者使用别人已设计好的类。C+标准本身并不提供现成的类的名称、结构和内容。在 C+中声明一个类类型和声明一个结构体类型是相似的。下面是声明一个结构体类型的方法:struct Student /声明了一个名为 Student 的结构体类型int num;char name20;char sex;Student stud1,stud2;/定义了两个结构体变量 stud1 和 stud2,它只包括数据,没有包括操作。现在声明一个类:class Student /以 c
17、lass 开头int num;char name20;char sex;/以上 3 行是数据成员void display( ) /这是成员函数coutnum:numendl;coutname:nameendl;coutsex:sexendl;/以上 4 行是函数中的操作语句;Student stud1,stud2;/定义了两个 Student 类的对象 stud1 和 stud2可以看到声明类的方法是由声明结构体类型的方法发展而来的。可以看到,类(class )就是对象的类型。实际上,类是一种广义的数据类型。类这种数据类型中的数据既包含数据,也包含操作数据的函数。不能把类中的全部成员与外界隔离
18、,一般是把数据隐蔽起来,而把成员函数作为对外界的接口。可以将上面类的声明改为class Student /声明类类型private : /声明以下部分为私有的int num;char name20;char sex;public : /声明以下部分为公用的void display( )coutnum:numendl;coutname:nameendl;coutsex:sexendl;Student stud1,stud2;/定义了两个 Student 类的对象如果在类的定义中既不指定 private,也不指定 public,则系统就默认为是私有的。归纳以上对类类型的声明,可得到其一般形式如下:
19、class 类名private : 私有的数据和成员函数;public : 公用的数据和成员函数;private 和 public 称为成员访问限定符(member access specifier)。除了private 和 public 之外,还有一种成员访问限定符 protected(受保护的),用protected 声明的成员称为受保护的成员,它不能被类外访问(这点与私有成员类似),但可以被派生类的成员函数访问。在声明类类型时,声明为 private 的成员和声明为 public 的成员的次序任意,既可以先出现 private 部分,也可以先出现 public 部分。如果在类体中既不写关
20、键字 private,又不写 public,就默认为 private。在一个类体中,关键字 private 和 public 可以分别出现多次。每个部分的有效范围到出现另一个访问限定符或类体结束时(最后一个右花括号)为止。但是为了使程序清晰,应该养成这样的习惯: 使每一种成员访问限定符在类定义体中只出现一次。在以前的 C+程序中,常先出现 private 部分,后出现 public 部分,如上面所示。现在的 C+ 程序多数先写 public 部分,把 private 部分放在类体的后部。这样可以使用户将注意力集中在能被外界调用的成员上,使阅读者的思路更清晰一些。在 C+程序中,经常可以看到类。
21、为了用户方便,常用的 C+编译系统往往向用户提供类库(但不属于 C+语言的组成部分),内装常用的基本的类,供用户使用。不少用户也把自己或本单位经常用到的类放在一个专门的类库中,需要用时直接调用,这样就减少了程序设计的工作量。三、定义对象的方法8.2.2节的程序段中,最后一行用已声明的 Student 类来定义对象,这种方法是很容易理解的。经过定义后,stud1和 stud2就成为具有 Student 类特征的对象。stud1和 stud2这两个对象都分别包括 Student 类中定义的数据和函数。定义对象也可以有以下几种方法: 先声明类类型,然后再定义对象前面用的就是这种方法,如 Studen
22、t stud1,stud2;/Student 是已经声明的类类型在 C+中,声明了类类型后,定义对象有两种形式。(1)class 类名 对象名如class Student stud1,stud2;把 class 和 Student 合起来作为一个类名,用来定义对象。(2)类名 对象名如Student stud1,stud2;直接用类名定义对象。这两种方法是等效的。第1种方法是从 C 语言继承下来的,第2种方法是 C+的特色,显然第2种方法更为简捷方便。 在声明类类型的同时定义对象class Student/声明类类型public :/先声明公用部分void display( )coutnum:
23、numendl;coutname:nameendl;coutsex:sexendl;private :/后声明私有部分int num;char name20;char sex;stud1,stud2;/定义了两个 Student 类的对象在定义 Student 类的同时,定义了两个 Student 类的对象。 不出现类名,直接定义对象class /无类名private : /声明以下部分为私有的public : /声明以下部分为公用的stud1,stud2;/定义了两个无类名的类对象直接定义对象,在 C+中是合法的、允许的,但却很少用,也不提倡用。在实际的程序开发中,一般都采用上面3种方法中的
24、第1种方法。在小型程序中或所声明的类只用于本程序时,也可以用第2种方法。在定义一个对象时,编译系统会为这个对象分配存储空间,以存放对象中的成员。四、类和结构体类型的异同C+增加了 class 类型后,仍保留了结构体类型(struct ),而且把它的功能也扩展了。C+允许用 struct 来定义一个类型。如可以将前面用关键字 class 声明的类类型改为用关键字 struct:struct Student /用关键字 struct 来声明一个类类型private : /声明以下部分为私有的int num;/以下 3 行为数据成员char name20;char sex;public: /声明以下
25、部分为公用的void display( ) /成员函数coutnum:numendl;coutname:nameendl;coutsex:sexendl;Student stud1,stud2;/定义了两个 Student 类的对象为了使结构体类型也具有封装的特征,C+不是简单地继承 C 的结构体,而是使它也具有类的特点,以便于用于面向对象程序设计。用 struct 声明的结构体类型实际上也就是类。用 struct 声明的类,如果对其成员不作 private 或public 的声明,系统将其默认为 public。如果想分别指定私有成员和公用成员,则应用 private 或 public 作显式
26、声明。而用 class 定义的类,如果不作 private 或 public 声明,系统将其成员默认为private,在需要时也可以自己用显式声明改变。如果希望成员是公用的,使用struct 比较方便,如果希望部分成员是私有的,宜用 class。建议尽量使用class 来建立类,写出完全体现 C+风格的程序。 6、C+类的成员函数(在类外定义成员函数、inline 成员函数)一、成员函数的性质类的成员函数(简称类函数)是函数的一种,它的用法和作用和前面介绍过的函数基本上是一样的,它也有返回值和函数类型,它与一般函数的区别只是:它是属于一个类的成员,出现在类体中。它可以被指定为 private(
27、私有的)、public (公用的)或 protected(受保护的)。在使用类函数时,要注意调用它的权限(它能否被调用)以及它的作用域(函数能使用什么范围中的数据和函数)。例如私有的成员函数只能被本类中的其它成员函数所调用,而不能被类外调用。成员函数可以访问本类中任何成员(包括私有的和公用的),可以引用在本作用域中有效的数据。一般的做法是将需要被外界调用的成员函数指定为 public,它们是类的对外接口。但应注意,并非要求把所有成员函数都指定为 public。有的函数并不是准备为外界调用的,而是为本类中的成员函数所调用的,就应该将它们指定为 private。这种函数的作用是支持其它函数的操作,
28、是类中其它成员的工具函数(utility function),类外用户不能调用这些私有的工具函数。类的成员函数是类体中十分重要的部分。如果一个类中不包含成员函数,就等同于 C 语言中的结构体了,体现不出类在面向对象程序设计中的作用。二、在类外定义成员函数在前面已经看到成员函数是在类体中定义的。也可以在类体中只写成员函数的声明,而在类的外面进行函数定义。如class Studentpublic :void display( );/公用成员函数原型声明private :int num;string name;char sex;/以上 3 行是私有数据成员;void Studentdisplay( )/在类外定义 display 类函数