C++面向对象程序设计课件_第1页
C++面向对象程序设计课件_第2页
C++面向对象程序设计课件_第3页
C++面向对象程序设计课件_第4页
C++面向对象程序设计课件_第5页
已阅读5页,还剩84页未读 继续免费阅读

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

返回本书目录

C++面向对象程序设计返回本章目录10.1.2C++语言概述10.1.1C++面向对象的程序实例10.1C++面向对象的程序实例10.1.1C++面向对象的程序实例返回本节目录2.C++程序实例(2)3.C++程序实例(3)1.C++程序实例(1)【例10.1】定义一个矩形类。(程序名为l10_1.cpp。)#include<iostream.h>classrectangle//定义一个矩形类{public:rectangle(floatlen,floatwid)//构造函数

{length=len; width=wid; } floatGetArea();//声明成员函数,计算矩形面积

floatGetPerimeter();//声明成员函数,计算矩形周长

~rectangle(){}//析构函数

private:floatlength;//私有数据

floatwidth;

};10.1.1C++面向对象的程序实例返回本节目录1.C++程序实例1floatrectangle::GetArea()//成员函数的具体实现{ returnlength*width;}floatrectangle::GetPerimeter()//成员函数的具体实现{ return2*(length+width);}voidmain(){floatl,w; cout<<"请输入矩形的长和宽:"; cin>>l>>w; rectanglex(l,w);//定义一个矩形类对象

cout<<x.GetArea()<<endl; cout<<x.GetPerimeter()<<endl;//调用成员函数}

10.1.1C++面向对象的程序实例返回本节目录10.1.1C++面向对象的程序实例返回本节目录实例1的功能是定义一个矩形类,该类有长和宽两个数据成员,用来描述矩形的静态特征(属性),有构造函数用来用初始化类对象,另外还有计算面积和周长两个成员函数作为该类的外部接口,供类外的程序访问。当用户输入矩形的长和宽之后,将构造一个实例矩形,并输出矩形的面积和周长。例如用户输入5,6<CR>,则输出结果为:

30

22【例10.2】类的派生。(程序名为l10_2.cpp。)

#include<iostream.h>classrectangle//定义矩形类{public:voidInitRect(floatlen,floatwid)//定义类的成员函数

{length=len; width=wid; } floatGetArea(); floatGetPerimeter();private://定义私有成员变量

floatlength; floatwidth;};floatrectangle::GetArea()//成员函数实现{ returnlength*width;}10.1.1C++面向对象的程序实例返回本节目录2.C++程序实例2floatrectangle::GetPerimeter()//成员函数实现{ return2*(length+width);}classsquare:publicrectangle//从矩形类中派生新类(正方形类){public: voidInitSquare(floatb){InitRect(b,b);}//新增的成员函数(初始化};//正方形)voidmain(){ squarex;//声明正方形类对象

x.InitSquare(8);//调用正方形类新增的成员函数

cout<<x.GetArea()<<endl;//调用从矩形类中继承下来的成员函数cout<<x.GetPerimeter()<<endl;//调用从矩形类中继承下来的成员}//函数(GetPerimeter)10.1.1C++面向对象的程序实例返回本节目录10.1.1C++面向对象的程序实例返回本节目录实例2的功能是先定义一个矩形类,然后从该矩形类中派生出一个新的正方形类(正方形是矩形的一个特例)。程序中先声明一个正方形类对象,然后将其初始化为边长为8的正方形,再调用从矩形类中继承下来的计算面积和周长两个函数,计算出正方形的面积和周长。该程序的输出结果为:64

32【例10.3】“+”运算符重载。(程序名为l10_3.cpp。)

#include<iostream.h>classrectangle//定义一个矩形类{public:rectangle(floatlen=0,floatwid=0)//构造函数

{ length=len; width=wid; } floatGetArea(){returnlength*width;}//成员函数(计算面积) rectangleoperator+(rectanglea2)//将"+"运算符重载

{rectanglea;//用于两个矩形对象相加

a.length=length; a.width=width+a2.GetArea()/length;returnrectangle(a.length,a.width); }10.1.1C++面向对象的程序实例返回本节目录3.C++程序实例3private://私有成员变量

floatlength; floatwidth;};voidmain(){rectanglex(5,9),y(5,6),z;//声明类对象

cout<<"第一个矩形面积为:"<<x.GetArea()<<endl; cout<<"第二个矩形面积为:"<<y.GetArea()<<endl;z=x+y;//对两个矩形相加

cout<<"两个矩形面积之和为:"<<z.GetArea()<<endl; }10.1.1C++面向对象的程序实例返回本节目录实例3的功能是先定义一个矩形类,然后将“+”运算符重载为可以使两个矩形类对象相加。该程序的输出结果为:

第一个矩形面积为:45

第二个矩形面积为:30

两个矩形面积之和为:7510.1.2C++语言概述返回本节目录

由于结构化程序设计自身的不足,在20世纪80年代出现了面向对象程序设计方法,C++语言也由此而产生。面向对象程序设计(Object-Orientedprogramming,简称OOP)设计的出发点就是为了能更直接的描述客观世界中存在的事物(即对象)以及它们之间的关系。面向对象程序设计是对结构化程序设计的继承和发展,它认为现实世界是由一系列彼此相关且能相互通信的实体组成,这些实体就是面向对象方法中的对象,而对一些对象的共性的抽象描述,就是面向对象方法中的类。类是面向对象程序设计的核心。10.1.2C++语言概述返回本节目录C++是目前最流行的面向对象程序设计语言。它在C语言的基础上进行了改进和扩充,并增加了面向对象程序设计的功能,更适合于开发大型的软件。C++是由贝尔实验室在C语言的基础开发成功的,C++保留了C语言原有的所有优点,同时与C语言又完全兼容。它既可以用于结构化程序设计,又可用于面向对象程序设计,因此C++是一个功能强大的混合型程序设计语言。

C++最有意义的方面是支持面向对象程序设计的特征。虽然与C语言的兼容性使得C++具有双重特点,但它在概念上和C语言是完全不同的,学习C++应该按照面向对象程序的思维去编写程序。返回本章目录10.2.2类的构造与封装10.2.1面向对象的基本概念10.2类与对象10.2.3创建对象10.2.4友元10.2.5模板10.2.6程序实训10.2.1面向对象的基本概念返回本节目录1.对象

从一般意义上讲,客观世界中任何一个事物都可以看成是一个对象。例如一本书,一名学生等。对象具有自己的静态特征和动态特征。静态特征可以用某种数据来描述,如一名学生的身高、年龄、性别等;动态特征是对象所表现的行为或具有的功能,如学生学习、运动、休息等。面向对象方法中的对象是系统中用来描述客观事物的一个实体,它是用来构成系统的一个基本单位,对象由一组属性和一组行为构成。属性是用来描述对象静态特征的数据项,行为是用来描述对象动态特征的操作序列。

许多对象具有相同的结构和特性,例如不管是数学书还是化学书,它们都具有大小、定价、编者等特性。在现实生活中,我们通常将具有相同性质的事物归纳、划分成一类,例如数学书和化学书都属于书这一类。同样在面向对象程序设计中也会采用这种方法。面向对象方法中的类是具有相同属性和行为的一组对象的集合。类代表了一组对象的共性和特征,类是对象的抽象,而对象是类的具体实例。例如,家具设计师按照家具的设计图做成一把椅子,那么设计图就好比是一个类,而做出来的椅子则是该类的一个对象,一个具体实例。拿【例10.1】中定义的矩形类来说,该类只是所有矩形的一个蓝本,它只是代表了矩形的一些特征,而该类的实例则是一个特定的矩形。10.2.1面向对象的基本概念返回本节目录2.类10.2.2类的构造与封装返回本节目录1.类的封装类的封装就是将对象的属性和行为结合成一个独立的实体,并尽可能隐蔽对象的内部细节,对外形成一道屏障,只保留有限的对外接口使之和外界发生联系。类的成员包括数据成员和成员函数,分别描述类所表达问题的属性和行为。对类成员的访问加以控制就形成了类的封装,这种控制是通过设置成员的访问权限来实现的。在面向对象程序设计中,通过封装将一部分行为作为外部接口,而将数据和其它行为进行有效的隐蔽,就可以达到对数据访问权限的合理控制。把整个程序中不同部分的相互影响减少到最低限度。10.2.2类的构造与封装返回本节目录2.类的定义类定义的一般格式为:class类名称

{

public:

公有数据和成员函数/*外部接口*/protected:

保护数据的成员函数private:

私有数据和成员函数

};10.2.2类的构造与封装返回本节目录关键字class说明了类定义的开始,类中所有的内容用大括号括起来。类的成员可分为三种级别的访问权限:public(公有的):说明该成员是公有的,它不但可以被类的成员函数访问,而且可以被外界访问,所以说公有类型定义了类的外部接口。Protected(保护的):说明该成员只能被该类的成员函数和该类的派生类的成员函数访问。Private(私有的):说明该成员只能被类的成员函数访问,外界不能直接访问它。类的数据成员一般都应该声明为私有成员。10.2.2类的构造与封装返回本节目录3.类的成员函数类的成员函数描述的是类的行为。定义类时在类定义体中给出函数的声明,说明函数的参数列表和返回值类型,而函数的具体实现一般在类定义体之外给出。下面是类外定义成员函数的一般格式:返回值类型类名::类成员函数名(参数列表){

函数体

}

其中::称为作用域分辨符。用它可以限制要访问的成员所在的类的名称。在建立一个对象时,常常需要作一些初始化工作,而当对象使用结束时,又需要作一些清理工作。在C++中提供了两个特殊的成员函数来完成这两项工作,那就是构造函数和析构函数。构造函数的作用就是在对象在被创建时利用特定的值构造对象,将对象初始化。构造函数完成的是一个从一般到具体的过程。需要注意的是构造函数同其它的成员函数不同,它不需要用户触发,而是在创建对象时由系统自动调用,其它任何过程都无法再调用构造函数。构造函数的函数名必须与类名相同,而且不能有返回值,也不需加void类型声明。构造函数可以带参数也可以不带参数。构造函数一般定义为公有类型。10.2.2类的构造与封装返回本节目录4.构造函数和析构函数10.2.2类的构造与封装返回本节目录析构函数也是类的一个公有成员函数,其作用与构造函数正好相反,它是用来在对象被删除前进行一些清理工作。析构函数调用之后,对象被撤消了,相应的内存空间也将被释放。析构函数名也应与类名相同,只是函数名前加一个波浪符“~”,以区别于构造函数。析构函数不能有任何参数,且不能有返回值。如果不进行显式说明,系统会自动生成缺省的析构函数,所以一些简单的类定义中没有显式的析构函数。5.构造函数与析构函数应用实例10.2.2类的构造与封装返回本节目录【例10.4】程序名为l10_4.cpp。

#include<iostream.h>

classA//定义类A

{public:

A()

{cout<<“构造函数被调用”<<endl;//构造函数

}

voiddisp()//成员函数

{cout<<“构造函数与析构函数应用举例”<<endl;

}

~A()//析构函数

{cout<<"析构函数被调用"<<endl;

}

};程序在声明A类的对象时,系统会自动调用构造函数,因而先执行构造函数中的输出语句,输出“构造函数被调用”,接下来调用disp成员函数,执行disp成员函数中的输出语句,输出“构造函数与析构函数应用举例”,最后程序在退出前由系统自动调用析构函数,执行析构函数中的输出语句,输出“析构函数被调用”,因此程序的输出结果为:

构造函数被调用构造函数与析构函数应用举例析构函数被调用10.2.2类的构造与封装返回本节目录voidmain()

{Aa;//声明类对象,自动调用构造函数

a.disp();//调用成员函数,对象使用结束时

//自动调用析构函数

}6.实例分析10.2.3创建对象返回本节目录

通过使用数组,我们可以对大量的数据和对象进行有效的管理,但对于许多程序,在运行之前,我们并不能确切地知道数组中会有多少个元素。例如,一个网络中有多少个可用节点,一个CAD系统中会用到多少个形状等。如果数组开的太大会造成很大的浪费,如果数组较小则又影响大量数据的处理。在C++中,动态内存分配技术可以保证我们在程序运行过程中按实际需要申请适量的内存,使用结束后再进行释放。这种在程序运行过程中申请和释放存储单元的过程称为创建对象和删除对象。1.new运算符C++用new运算符来创建对象。

运算符new的功能是动态创建对象,其语法形式为:

new类型名(初值列表);

该语句的功能是在程序运行过程中申请用于存放指定类型的内存空间,并用初值列表中给出的值进行初始化。

如果创建的对象是普通变量,初始化工作就是赋值,如果创建的对象是某一个类的实例对象,则要根据实际情况调用该类的构造函数。

例如:int*p;

p=newint(100);//赋值给指针变量

例如:rectangle*r;

r=newrectangle(5,6);//调用矩形类的构造函数

如果创建的对象是数组类型,则应按照数组的结构来创建,其语法形式为:10.2.3创建对象返回本节目录10.2.3创建对象返回本节目录一维数组:new类型名[下标];

当数组为一维数组时,下标为数组为元素的个数,动态分配内存时不能指定数组元素的初值。如果内存申请成功,返回一个指向新分配内存的首地址的指定类型的指针,如果失败返回0。

例如:int*p;

p=newint[10];

多维数组:new类型名[下标1][下标2]……;

当数组为多维数组时,返回一个指向新分配内存的首地址的指针,但该指针的类型为指定类型的数组。数组元素的个数为除最左边一维外下标的乘积。

例如:int(*p)[5];

p=newint[10][5];10.2.3创建对象返回本节目录2.delete运算符运算符delete的功能是删除由new运算符创建的对象,释放指针指向的内存空间,其语法形式为:

delete指针名;

例如:int*p;

p=newint(100);

deletep;

如果new运算符创建的对象是数组,则删除对象时应使用的语法形式为:

delete[]指针名;

例如:int*p;

p=newint[10];

delete[]p;10.2.3创建对象返回本节目录3.C++程序实例【例10.5】类对象的创建与删除。(程序名为l10_5.cpp。)

#include<iostream.h>

classrectangle//定义一个矩形类{public:

rectangle(floatlen,floatwid)//构造函数

{length=len;

width=wid;

}

floatGetArea();//成员函数

private:

floatlength;

floatwidth;

};10.2.3创建对象返回本节目录floatrectangle::GetArea()//成员函数实现{returnlength*width;

}

voidmain()

{rectangle*r;//定义指向rectangle类的指针变量r

r=newrectangle(10,5);//创建rectangle类对象

cout<<r->GetArea()<<endl;

deleter;//删除对象

}4.C++程序实例分析程序中先建立了一个rectangle类,然后在主函数中定义了一个指向rectangle类的指针变量r,用new在内存中开辟一段空间以存放rectangle类对象,这时会自动调用构造函数来初始化该对象,接下来使用指向rectangle类的指针变量r得到矩形的面积,最后用delete删除对象,释放该空间。10.2.4友元返回本节目录2.友元函数3.友元类1.友元概述10.2.4友元返回本节目录

在程序设计过程中,假如用户建立了一个类,这个类的数据成员被定义为私有,这时如果想把该数据存取到某个函数(非该类的成员函数)中,那么这样作肯定是不被允许的。但是有时候一些非成员函数需要亲自访问类中的某些私有数据,那么这时候就需要用到友元。友元提供了不同类或对象的成员函数之间、类的成员函数与一般函数之间进行数据共享的机制。通过友元,一个普通函数或类的成员函数可以访问到封装于其它类中的数据。友元的作用在于提高程序的运行效率,但同时它也破坏了类的封装性和隐藏性,使得非成员函数可以访问类的私有成员。1.友元概述10.2.4友元返回本节目录

友元函数是在类定义中由关键字friend修饰的非成员函数。友元函数可以是一个普通函数,也可以其它类中的一个成员函数,它不是本类的成员函数,但它可以访问本类的私有成员和保护成员。友元函数需要在类的内部进行声明。其格式如下:

class类名称

{

类的成员声明

friend返回类型友元函数名(参数列表);

类的成员声明

}2.友元函数10.2.4友元返回本节目录【例10.6】使用友元函数计算两点间的距离。程序名为l10_6.cpp。#include<iostream.h>

#include<math.h>

classPoint//定义Point类(点)

{public:

Point(doublexx,doubleyy)//构造函数

{x=xx;

y=yy;}

voidGetxy();//成员函数声明

frienddoubleDistance(Point&a,Point&b);//友元函数声明

private:

doublex,y;//私有成员(点的坐标)

};voidPoint::Getxy()//成员函数实现{cout<<“(”<<x<<“,”<<y<<“)”;//输出点的坐标

}

doubleDistance(Point&a,Point&b)//友元函数实现

{doubledx=a.x-b.x;//访问私有成员

doubledy=a.y-b.y;

returnsqrt(dx*dx+dy*dy);//两点距离

}

voidmain()

{Pointp1(3.0,4.0),p2(6.0,8.0);//定义Point类对象

p1.Getxy();//调用成员函数

p2.Getxy();

cout<<“\nTheistanceis”<<Distance(p1,p2)<<endl;

//调用友元函数

}

10.2.4友元返回本节目录10.2.4友元返回本节目录实例说明:程序在类定义体内声明了友元函数的原形,在声明时函数名前加friend关键字进行修饰,友元函数的具体实现在类外给出,可以看出友元函数通过对象名可直接访问Point类的私有成员,而不需要借用外部接口Getxy。在调用友元函数时同调用普通函数一样,采用直接调用方法,而不需要像调用类的成员函数那样必须通过对象。

该程序的输出结果为:(3,4)(6,8)Thedistanceis510.2.4友元返回本节目录3.友元类同函数一样,类也可以声明为另一个类的友元,这时称为友元类。当一个类作为另一个类的友元时,这个类的所有成员函数都将是另一个类的友元函数。友元类的一般格式为:class类A{

类A的成员声明

friendclass类B;

类A的成员声明}【例10.7】程序名为l10_7.cpp。#include<iostream.h>classA//定义类A{public: A(intxx,intyy)//类A的构造函数

{ x=xx; y=yy;

} friendclassB;//声明类B为类A的友元类private:

intx,y;};classB//定义类B{public: voiddisp1(As){cout<<"disp1调用了类A的私有成员x:"<<s.x<<endl;} //类B的成员函数访问类A的私有成员

10.2.4友元返回本节目录程序中定义了A和B两个类,其中类B为类A的友元类,定义时在类A的内部声明类B,而类B的具体实现过程是在类A的外部。类B中有两个成员函数disp1和disp2,根据友元类的概念,这两个成员函数都成为类A的友元函数,所以它们都可以访问类A的私有成员x和y。该程序的输出结果为:disp1调用了类A的私有成员x:5disp2调用了类A的私有成员y:910.2.4友元返回本节目录 voiddisp2(As) {cout<<"disp2调用了类A的私有成员y:"<<s.y<<endl;} //类B的成员函数访问类A的私有成员

};

voidmain(){Aa(5,9);//声明A类对象

Bb;//声明B类对象

b.disp1(a);//调用B类的成员函数

b.disp2(a);}10.2.5模板返回本节目录前面章节中我们学习了重载函数,使用重载函数可以处理多种数据类型,但是即使设计为重载函数也只是使用相同的函数名,函数体仍然要分别定义。使用函数模板则不同了,函数模板可以用来创建一个通用功能的函数,以支持多种不同的参数,简化重载函数的函数体设计。函数模板的定义形式为:

template<typename标识符>

函数定义1.函数模板10.2.5模板返回本节目录【例10.8】使用函数模板。(程序名为l10_8.cpp。)#include<iostream.h>template<typenameT>Tmax(Tx,Ty)//定义模板函数{returnx>y?x:y;}voidmain(){inta=6,b=8;cout<<max(a,b)<<endl;doublem=10.5,n=8.5;cout<<max(m,n)<<endl;}程序中首先定义模板函数max用来取得两个数中相对较大的一个,然后主函数调用该函数,此时编译器将根据实参的类型推导出函数模板的类型参数。例如,当调用max(a,b)时,由于a,b为整型变量,所以模板中类型参数T为int,而调用max(m,n)时,由于m,n为double类型,所以模板中类型参数T为double。10.2.5模板返回本节目录2.类模板

使用类模板使用户可以为类定义一种模式,使得类中的某些数据成员、某些成员函数的参数、某些成员函数的返回值能取任意类型,包括用户自定义的或系统预定义的。

为了定义类模板,应在类的声明之前加上一个模板参数表,参数表里的形式类型名用来说明成员数据和成员函数的类型。类模板的定义形式为:

template<class类型说明符>

类定义

类模板自身不产生代码,它指定类的一个家族,当被其它代码引用时,类模板才根据引用的需要产生代码。【例10.9】使用类模板。(程序名为l10_9.cpp。)

#include<iostream.h>

template<classT>//类模板,实现对任意数据类型的存取classrectangle//定义一个矩形类rectangle

{public:

rectangle(Tlen,Twid)//构造函数

{length=len;

width=wid;

}

TGetArea();//声明成员函数,计算矩形面积

TGetPerimeter();//声明成员函数,计算矩形周长private:

Tlength;//私有数据,用于任意类型

Twidth;

};

template<classT>

Trectangle<T>::GetArea()//成员函数的具体实现

{returnlength*width;}10.2.5模板返回本节目录template<classT>

Trectangle<T>::GetPerimeter()//成员函数的具体实现

{return2*(length+width);}

voidmain()

{rectangle<int>x(4,6);//定义rectangle对象(成员函数返回值为int型)

rectangle<double>y(5.5,3.8);//定义rectangle对象(成员函数返回值为double型)

cout<<“x的面积是:”<<x.GetArea()<<endl;

cout<<“x的周长是:”<<x.GetPerimeter()<<endl;

cout<<“y的面积是:"<<y.GetArea()<<endl;

cout<<"y的周长是:"<<y.GetPerimeter()<<endl;

}程序的执行结果为:

x的面积是:24

x的周长是:20

y的面积是:20.9

y的周长是:18.610.2.5模板返回本节目录【例10.10】日期类应用程序举例。(程序名为l10_10.cpp。)

#include<iostream.h>

classTDate//日期类的定义

{public:

TDate(inty,intm,intd);//构造函数

intIsLeapYear();//判断闰年

voidshow();//显示日期

friendintGetyear(TDatedate);//友元函数,返回年

private:

intyear,month,day;

};

TDate::TDate(inty,intm,intd)//构造函数实现

{year=y;

month=m;

day=d;

}10.2.6程序实训返回本节目录1.C++程序实例intTDate::IsLeapYear()

{return(year%4==0&&year%100!=0)||(year%400==0);}

voidTDate::show()

{cout<<year<<“年”<<month<<“月”<<day<<“日”<<endl;}

intGetyear(TDatedate)//友元函数实现

{returndate.year;}

voidmain()

{TDate*date;//声明指向TDate类型的指针变量

date=newTDate(2008,10,28);//创建对象

date->show();//调用成员函数

if(date->IsLeapYear())

cout<<Getyear(*date)<<“年是闰年”<<endl;//调用友元函数

else

cout<<Getyear(*date)<<“年不是闰年"<<endl;

deletedate;//删除对象

}10.2.6程序实训返回本节目录10.2.6程序实训返回本节目录程序可以分为三个部分,第一部分是CDate类的定义,第二部分是成员函数的实现,第三部分是主函数。类定义中仍然是由成员数据和成员函数组成,其中成员函数作为外部接口。程序中为了说明友元的应用,引入了友元函数,这里可以换成类的成员函数,读者可以自己试一试。主函数中,在定义了指向TDate类型的指针变量之后,使用new运算符创建对象,程序结束时又使用delete运算符删除了对象,释放了内存空间。这种编程方法在以后的程序设计中会经常遇到,读者应该习惯于使用new和delete进行动态内存管理。2.C++程序实例分析返回本章目录10.3.2派生类10.3.1

继承与派生的概念10.3继承与派生10.3.4虚函数10.3.3

派生类的构造函数与析构函数10.3.5程序实训

面向对象程序设计过程中,一个很重要的特点就是代码的可重用性。C++是通过继承这一机制来实现代码重用的。所谓继承指的是类的继承,就是在原有类的基础上建立一个新类,新类将从原有类那里得到已有的特性。例如,现有一个学生类,定义了学号、姓名、性别、年龄四项内容,如果我们除了用到以上四项内容外,还需要用到电话和地址等信息,我们就可以在学生类的基础上再增加相关的内容,而不必重头再定义一个类。换个角度来说,从已有类产生新类的过程就是类的派生。类的继承与派生允许在原有类的基础上进行更具体、更详细的修改和扩充。新的类由原有的类产生,包含了原有类的关键特征,同时也加入了自己所特有的性质。新类继承了原有类,原有类派生出新类。原有的类我们称为基类或父类,而派生出的类我们称为派生类或子类。比如:所有的windows应用程序都有一个窗口,可以说它们都是从一个窗口类中派生出来的,只不过有的应用程序用于文字处理,有的则应用于图像显示。10.3.1继承与派生的概念返回本节目录10.3.1继承与派生的概念返回本节目录

类的继承与派生的层次结构是人们对自然界中事物进行分类分析认识过程在程序设计中的体现。现实世界中的事物都是相互联系、相互作用的,人们在认识过程中,根据它们的实际特征,抓住其共同特性和细小差别,利用分类的方法进行分析和描述。

如下图是生物类的一种继承图,最高层次的生物类是抽象程度最高的,是最具有普遍意义的概念,下层具有上层的特性,同时也加入了自己新的特性,而最下层是最具体的。这个层次结构中,由上到下,是一个具体化、特殊化的过程,由下到上,是一个抽象化的过程。上下层之间就可以看作是基类与派生类的关系。10.3.1继承与派生的概念返回本节目录10.3.2派生类返回本节目录定义派生类的一般形式为:

class派生类名:继承方式基类名

{派生类成员定义};

继承方式指派生类的访问控制方式,可以是public(公有继承),protected(保护继承)和private(私有继承)。继承方式可以省略不写,缺省值为private。

派生类在派生过程中,需要经历吸收基类成员、改造基类成员和添加新成员三个步骤。其中,吸收基类成员主要目的是实现代码重用,但应该注意的是基类的构造函数和析构函数是不能被继承下来的;改造基类成员则主要是对基类数据成员或成员函数的覆盖,就是在派生时定义一个和基类数据或函数同名的成员,这样基类中的成员就被替换为派生类中的同名成员;添加新成员是对类功能的扩展,添加必要的数据成员和成员函数来实现新增功能。10.3.2派生类返回本节目录

在继承中,派生类继承了除构造函数和析构函数以外的全部成员。对于从基类继承过来的成员的访问并不是简单的把基类的私有成员、保护成员和公有成员直接作为私有成员、保护成员和公有成员,而是要根据基类成员的访问权限和派生类的继承方式共同决定。

1.公有继承

当类的继承方式为public时,基类的公有(public)成员和保护(protected)成员仍然成为派生类的公有成员和保护成员,而基类的私有成员不能被派生类访问。

2.保护继承

当类的继承方式为protected时,基类的公有(public)成员和保护(protected)成员将成为派生类的保护成员,而基类的私有成员不能被派生类访问。10.3.2派生类3.私有继承

当类的继承方式为private时,基类的公有(public)成员和保护(protected)成员将成为派生类的私有成员,而基类的私有成员不能被派生类访问。(看下图:)返回本节目录如10.1.1中实例2先定义了一个矩形类,然后又从矩形类中派生出正方形类,正方形是矩形的一个特例,矩形类实例化一个矩形对象时需要用到长和宽两个参数,而正方形则只需要用到边长一个参数,所以在派生过程中新增了成员函数InitSquare用来完成初始化工作。InitSquare函数实现过程中用到了基类的成员函数InitRect,因为InitRect函数为基类的公有成员,所以在派生类中可以被访问。如果InitSquare函数的实现过程改为如下语句:

voidInitSquare(floatb)

{length=b;

width=b;}

该程序是不能被编译通过的,因为派生类是不能访问基类的私有成员的。另外,该程序使用的继承方式为公有继承,因此基类中的公有成员都被派生类吸收并成为公有成员。10.3.2派生类返回本节目录10.3.3派生类的构造函数与析构函数返回本节目录

继承的目的是为了发展,派生类继承了基类的成员,实现了代码重用,这只是一部分,而代码的扩充才是最主要的,只有添加新的成员,加入新的功能,类的派生才有实际意义。我们知道,基类的构造函数和析构函数是不能被继承的,因此如果要对新增成员进行初始化,就必须加入新的构造函数。同样,对派生类对象的扫尾、清理工作也需要引入新的析构函数。1.派生类的构造函数如果基类的对象包含对成员的初始化,而在建立派生类对象时,由于基类的构造函数不能被继承而无法执行,因此会使基类的成员未初始化,所以在设计派生类的构造函数10.3.3派生类的构造函数与析构函数返回本节目录时,不仅要考虑派生类新增的数据成员的初始化,还应当考虑基类数据成员的初始化。

派生类构造函数的一般格式为:

派生类构造函数名(参数列表):基类构造函数名(参数列表)

{

派生类新增成员初始化语句;

}

将【例10.2】改为使用构造函数进行初始化:【例10.11】程序名为l10_11.cpp。

#include<iostream.h>

classrectangle//定义矩形类(基类)

{public:

rectangle(floatlen,floatwid)//基类的构造函数 {length=len;

width=wid;

}floatGetArea();

floatGetPerimeter();

private://定义私有成员变量

floatlength;

floatwidth;

};

floatrectangle::GetArea()//基类成员函数实现

{returnlength*width;

}

floatrectangle::GetPerimeter()//基类成员函数实现

{return2*(length+width);

}

classrect:publicrectangle//从矩形类中派生新类(正方形类)

{public:

rect(floatb):rectangle(b,b)//派生类的构造函数

{}

};10.3.3派生类的构造函数与析构函数返回本节目录10.3.3派生类的构造函数与析构函数返回本节目录voidmain()

{rectx(8);//声明正方形类对象

cout<<x.GetArea()<<endl;//调用从矩形类中继承

//下来的成员函数(GetArea)

cout<<x.GetPerimeter()<<endl;//调用从矩形类中继承下来

//的成员函数(GetPerimeter)

}程序的输出结果为:64

3210.3.3派生类的构造函数与析构函数返回本节目录2.派生类的析构函数在派生过程中,基类的析构函数也不能被继承下来,这就需要在派生类中自行定义。派生类析构函数的定义与没有继承关系的类的析构函数完全相同,只要负责把派生类新增的成员的清理工作做好就够了,系统会自动调用基类的析构函数对基类对象成员进行清理。同构造函数相同,析构函数在执行过程中也要对基类进行操作,它首先对派生类新增成员进行清理,然后对基类进行清理。这些清理工作分别是调用派生类析构函数和调用基类析构函数来完成的。

【例10.12】程序名为l10_12.cpp。

#include<iostream.h>

classcolor//定义color类

{public:

voidpaint(){cout<<“Nocolor.”<<endl;}

};

classgreen:publiccolor//由color派生出green类

{public:

voidpaint(){cout<<"Thecolorisgreen."<<endl;}

};10.3.4虚函数返回本节目录

虚函数就是在前面使用virtual关键字来限定的普通成员函数。虚函数是为实现某种功能而假设的函数,它是类的一个成员函数。当在派生类中定义了一个同名的成员函数时,只要该成员函数的参数类型及返回类型与基类中同名的虚函数完全一样,那么无论是否用virtual关键字,它都将成为一个虚函数。

我们先看一个例子:10.3.4虚函数返回本节目录voidmain()

{ colorc,*p;//声明color类变量及指针变量

greeng;//声明green类变量

p=&g;//将指向color类的指针p指向green类

c.paint();

g.paint();

p->paint();

}程序执行结果如右图:10.3.4虚函数程序中对象c和对象g的输出结果都不难理解,但指针p已经指向了green类的对象,然而从输出结果看调用的却是color类的paint()函数。解决这个问题的方法就是使用虚函数。

将上例color类改为如下使用虚函数的方法:classcolor//定义color类

{public:

virtualvoidpaint(){cout<<"Nocolor."<<endl;}//定义虚函数

};输出结果如右图:返回本节目录10.3.4虚函数返回本节目录

以上实例说明在C++中,如果一个函数被定义成虚函数,那么,即使是使用指向基类对象的指针来调用该成员函数,C++也能保证所调用的是正确的特定于实际对象的成员函数。10.3.5程序实训返回本节目录

使用类的派生编程实现在屏幕上显示某学生的选课信息。学生包括姓名和学号两个属性,选课信息包括课程名称、任课教师和上课时间三个属性,每名学生最多可选三门课。由于学生包括两个属性,选课信息包括三个属性,而这些属性都可以用字符数组表示,那么抓住这个共性我们就可以考虑建立一个基类(base),使其具有两个字符数组类型的成员变量,接下来我们就可以由该基类派生出学生类(student)和选课信息类(classes)。为了显示相关信息,在派生的过程中,需要增加相应的外部接口。程序的具体实现代码如下:10.3.5程序实训【例10.14】程序名为l10_14.cpp。#include<iostream.h>

#include<string.h>

classbase//定义基类base

{private:

charstr1[20];

charstr2[10];

public:

base(){}

base(chars1[],chars2[])

{strcpy(str1,s1);

strcpy(str2,s2);

}

voidshow1()

{cout<<str1;

}返回本节目录10.3.5程序实训返回本节目录voidshow2()

{cout<<str2;

}

};

classclasses:publicbase//从基类派生课程类

{private:

charcname[20];//课程名称

public:

classes(){}

classes(charcna[],charcweek[],charctea[]):base(cweek,ctea)

//课程名称、上课时间、任课教师

{strcpy(cname,cna);}voidshow()

{cout<<cname<<“(”;

show1();//课程名称(任课教师,上课时间)

cout<<“,”;

show2();

cout<<“)”;

}

};

classstudent:publicbase//从基类派生学生类

{private:

classescls[3];//每名学生最多选修3门课

intmax;

public:

student(){}

student(charsname[],charsno[]):base(sname,sno){max=0;}10.3.5程序实训返回本节目录10.3.5程序实训返回本节目录voidselectcls(classesc)//选课

{ cls[max]=c;

max++;

}

voidshowcls()//显示选课信息

{cout<<“学生:”;

show1();

cout<<"(";

show2();

cout<<")"<<endl<<"选修课程:"<<endl;

for(inti=0;i<max;i++)

{cout<<i+1<<":";

cls[i].show();

cout<<endl;

}

}

};10.3.5程序实训返回本节目录voidmain(){studentstu("黄雪菲","12345678");classescls1("数据库应用","曲子研","星期三");classescls2("C++程序设计","胡小燕","星期二");classescls3("数据结构","江山","星期五");stu.selectcls(cls1);stu.selectcls(cls2);stu.selectcls(cls3);stu.showcls(); }程序的执行结果如右图:返回本章目录10.4.2运算符重载规则10.4.1重载运算符10.4运算符重载10.4.3类型转换与转换函数10.4.1重载运算符返回本节目录

运算符重载增强了C++语言的可扩充性。在C++中,定义一个类就是定义了一个新类型。因此,类对象和变量一样,可以作为参数传递,也可以作为返回类型。在基本数据类型中,系统提供了许多预定义的运算符,而实际上,对于用户自定义的类型(比如类),也需要有类似的运算操作,这就提出了对运算符进行重新定义,赋于已有符号以新的功能要求。运算符重载的实质是函数重载。所谓函数重载简单地说就是赋给同一个函数名多个含义,C++中允许在相同的作用域内以相同的名字定义几个不同的函数,可以是成员函数,也可以是非成员函数,定义重载函数时要求函数的参数或者至少有一个类型不同,或者个数不同,而对于返回值的类型可以相同,也可以不同。对于每一个运算符@,都对应于一个10.4.1重载运算符返回本节目录运算符函数operator@,例如“+”对应的运算符函数是operator+。运算符重载在实现过程中首先把指定的运算符转化为对运算符函数的调用,再把运算对象转化为运算符函数的实参,然后根据实参的类型来确定需要调用的运算符函数。

例如在10.1.1实例3中,我们把“+”重载为矩形类的成员函数,那么就可直接进行两个矩形类对象相加了。可以看出,运算符重载除了在函数声明时使用了关键字operatore之外,其它地方与类的普通成员函数没什么区别,只不过它可以直接通过运算符和操作数的方式来完成函数的调用。运算符重载扩充了运算符原有的功能,如10.1.1实例3中对“+”进行了运算符重载之后,“+”的原有功能不变,对整型、浮点型数据的运算仍然遵循C++预定义的规则,同时又增加了针对矩形类运算的功能。“+”作用于不同的对象上,就会导致不同的操作行为,具有了更广泛的功能。10.4.2运算符重载规则返回本节目录运算符重载的规则如下:

1)C++中的运算符除了少数几个外,全部可以重载,而且只能重载已有的这些运算符。

2)重载之后运算符的优先级和结合性保持不变。

3)运算符重载是针对新类型数据的实际需要,对原有运算符进行适当的改造。一般来讲,重载的功能应与原有的功能相类似,不能改变原有运算符的操作对象个数,同时至少有一个操作对象是自定义类型。

不能重载的运算符有五个,它们是是类属关系运算符“.”、指针运算符“*”、作用域运算符“::”、三目运算符“?:”和sizeof运算符。

运算符的重载形式有两种,重载为类的成员函数和重载为友元函数。重载为类的成员函数的一般形式为:返回本节目录

类名称operator运算符(参数表)

{

函数体

}

重载为友元函数的一般形式为:

friend函数类型operator运算符(参数表)

{

函数体

}10.4.2运算符重载规则10.4.2运算符重载规则返回本节目录

函数类型指定了重载运算符的返回值类型,operator是定义运算符重载函数的关键字,运算符是C++中可重载的运算符。当运算符重载为类的成员函数时,双目运算符仅有一个参数,而单目运算符则不能显式的说明参数。一般情况下,单目运算符最好重载为成员函数,双目运算符最好重载为友元函数,双目运算符重载为友元函数比重载为成员函数更方便操作,但是,有的双目运算符重载为成员函数为好,例如,赋值运算符。如果被重载为友元函数,将会出现与赋值语义不一致的地方。

在C++中,如果编译器遇到一个表达式或函数的调用使用了一个不合适的数据类型,它经常会类型转换。类型转换是将一种类型的值映射为另一种类型的值。类型转换可分为隐式转换和显式转换两种。C++编译系统提供的数据类型的隐式转换规则如下:(1)程序在执行算术运算时,低类型可以转换为高类型。

(2)在赋值表达式中,右边表达式的值自动转换为左边变量的类型,并赋值给它。

(3)在函数调用时,将实参值赋给形参,系统隐式地将实参转换为形参的类型后赋给形参。

(4)函数有返回值时,系统将自动地将返回的表达式类型转换为函数返回值类型并赋值给该函数。

10.4.3类型转换与转换函数返回本节目录1.类型转换10.4.3类型转换与转换函数返回本节目录当在程序中发现两个数据类型不相容时,又不能自动完成隐式转换,则将出现编译错误。

如:int*p=10;

在这种情况下,为了消除错误,可以进行如下所示的显示类型转换:

int*p=(int*)10;//将整型数10显式地转换成指针类型2.转换函数在C++中,可以通过定义类型转换函数来为用户定义类型达到相同的效果。这些函数有特殊类型的构造函数和重载的运算符。10.4.3类型转换与转换函数返回本节目录1)构造函数转换在实际应用中,当类定义中提供了单个参数的构造函数时,该类便提供了一种将其他数据类型的数值或变量转换为用户所定义数据类型的方法。因此,可以说单个参数的构造函数提供了数据转换的功能。请看下面的例子:【例10.15】程序名为l10_15.cpp。

#include<iostream.h>

classA//定义类A

{public:

A(doublex){m=x;}

voidprint(){cout<<m<<endl;}

private:

doublem;

};10.4.3类型转换与转换函数返回本节目录voidmain()

{Aa(5);

a=100;//给A类类型的对象赋值

a.print();

}

在该程序中,赋值语句a=100,赋值号两边分别是数值100和对象a,两者是不相容的数据类型,可是它却能顺利通过编译程序,并且输出显示正确结果,其主要原因是因为单参数的构造函数。编译系统首先将整型数据100转换成double型,然后,再通过类中定义的单参数构造函数将double型数值转换为A类类型,最后把它赋值给a。这些转换都是自动隐式完成的。10.4.3类型转换与转换函数返回本节目录2)运算符转换函数

运算符转换方法是通过运算符重载实现的。用户可以创建一个成员

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

评论

0/150

提交评论