C++及Windows可视化程序设计第5章.pptx

上传人:莉*** 文档编号:87569049 上传时间:2023-04-16 格式:PPTX 页数:177 大小:698.06KB
返回 下载 相关 举报
C++及Windows可视化程序设计第5章.pptx_第1页
第1页 / 共177页
C++及Windows可视化程序设计第5章.pptx_第2页
第2页 / 共177页
点击查看更多>>
资源描述

《C++及Windows可视化程序设计第5章.pptx》由会员分享,可在线阅读,更多相关《C++及Windows可视化程序设计第5章.pptx(177页珍藏版)》请在taowenge.com淘文阁网|工程机械CAD图纸|机械工程制图|CAD装配图下载|SolidWorks_CaTia_CAD_UG_PROE_设计图分享下载上搜索。

1、5.7 重载对象的赋值运算符5.8 对象成员的初始化5.9 类模板与标准模板库5.10 面向对象编程的文件规范实验习题第1页/共177页本章重点介绍在C+中定义类、建立和使用对象的方法。虽然同类对象在其数据成员的取值方面是不相同的,但可以共用相同的代码。类是对同类对象的描述,它不但描述了对象之间的公有接口,同时也给出了对象的内部实现(数据成员和成员函数)。第2页/共177页像构造枚举和结构一样,类也是一种用户自己构造的数据类型并遵循C+的规定。例如,类也要先声明后使用;不管声明的内容是否相同,声明同一个名字的两个类是错误的,类是具有惟一标识符的实体;在类中声明的任何成员不能使用extern、a

2、uto和register关键字进行修饰;类中声明的变量属于该类,在某些情况下,变量可以被该类的不同实例所共享。5.1 类及其实例化 5.1.1 定义类第3页/共177页类和其他数据类型不同的是,组成这种类型的不仅可以有数据,而且可以有对数据进行操作的函数,它们分别叫做类的数据成员和类的成员函数,而且不能在类声明中对数据成员使用表达式进行初始化。第4页/共177页1.声明类类是对一组性质相同对象的程序描述。在C+中声明类的一般形式为 class 类名 private:私有数据和函数 public:公有数据和函数 protected:保护数据和函数;第5页/共177页类声明以关键字class开始,

3、其后跟类名。类所声明的内容用花括号括起来,右花括号后的分号作为类声明语句的结束标志。这一对花括号“”之间的内容称为类体。类中定义的数据和函数称为这个类的成员(数据成员和成员函数)。类成员均具有一个属性,叫做访问权限,通过它前面的关键字来定义。顾名思义,关键字private、public和protected 以后的成员的访问权限分别是私有、公有和保护的,把这些成员分别叫做私有成员、公有成员和保护成员。访问权限用于控制对象的某个成员在程序中的可访问性,如果没有使用关键字,则所有成员默认声明为private权限。这些关键字的使用顺序和次数也都是任意的。第6页/共177页【例5.1】描述点的Point

4、类。class Point /类名Pointprivate:/声明为私有访问权限 int x,y;/私有数据成员public:/声明为公有访问权限 void Setxy(int a,int b);/无返回值的公有成员函数 void Move(int a,int b);/无返回值的公有成员函数 void Display();/无返回值的公有成员函数 int Getx();/返回值为int的公有成员函数 int Gety();/返回值为int的公有成员函数;/类声明以分号结束x和y是私有成员,Setxy、Display、Move、Getx和Gety是公有成员。因为只是声明函数,所以可只给出函数原型

5、。【例5.2】是其等效的声明方式。第7页/共177页【例5.2】使用默认关键字及改变关键字顺序和次数的Point类。class Point /类名Pointint x;/默认私有数据成员 public:/声明为公有访问权限 /无返回值的公有成员函数Setxy的函数原型 void Setxy(int,int);/无返回值的公有成员函数Move的函数原型void Move(int,int);void Display();/无返回值的公有成员函数的函数原型int Getx();/返回值为int的公有成员函数的函数原型int Gety();/返回值为int的公有成员函数的函数原型第8页/共177页pr

6、ivate:/声明为私有访问权限int y;/私有数据成员;/类定义以分号结束由此可见,成员函数声明的规则与第4章所述的函数声明规则相同。2.定义成员函数类中说明的成员函数用来对数据成员进行操作。例如,Point类的Setxy函数用来为该类的对象设置初始值,而当调用成员函数Getx时,则返回一个对象的数据成员x的值。在类中只对这些成员函数进行了函数声明,还必须在程序中实现这些成员函数。定义成员函数的一般形式为第9页/共177页 返回类型 类名成员函数名(参数列表)成员函数的函数体/内部实现 其中“”是作用域运算符,“类名”是成员函数所属类的名字,“”用于表明其后的成员函数是属于这个特定的类。换

7、言之,“类名成员函数名”的意思就是对属于“类名”的成员函数进行定义,而“返回类型”则是这个成员函数返回值的类型。余下的工作就是定义成员函数的函数体。例如Setxy是类Point的成员函数,它没有返回值,则定义如下:void Point Setxy(int a,int b)x=a;y=b;第10页/共177页将“void PointSetxy(int a,int b)”理解为定义Point的函数成员Setxy(int a,int b),该成员带有两个整型参数,函数没有返回值(void)。按此方法,可写出其他几个成员函数的定义:void Point Move(int a,int b)x=x+a;y

8、=y+b;void Point Display()coutx,yendl;int Point Getx()return x;int Point Gety()return y;第11页/共177页也可以使用关键字inline将成员函数定义为内联函数,例如:inline int Point Getx()return x;如果在声明类的同时,在类体内给出成员函数的定义,则默认为内联函数。例如在类中将声明Getx的语句“int Getx();”改为“int Getx()return x;”,则Getx为内联函数。一般直接在类体内给出简单成员函数的定义。有些成员函数的实现方式不止一种,例如void Po

9、int Display()coutGetx(),Gety()endl;第12页/共177页是调用成员函数Getx()和Gety()实现的,它们使用了cout流,应在定义之前包含如下语句:#include using namespace std;3.数据成员的赋值不能在类体内给数据成员赋值,即下面的方法是错误的:Class Point int x=25,y=56;第13页/共177页当然,在类体外面就更不允许了。数据成员的具体值是用来描述对象的属性的。只有产生了一个具体的对象,这些数据值才有意义,所以又称对象的初值或对象初始化。这跟简单数据类型的道理一样,int是整数类型,但需要声明整数对象之后

10、才有意义,“int x=5;”使得整数对象x的值为5,只是类具有成员函数而已。假设已经有了一个对象A,则可使用“.”运算符调用成员函数Setxy赋初值。例如:A.Setxy(25,56);第14页/共177页将对象A的数据成员x和y分别赋给25和56,即A.x=25,A.y=56。其实,真正的初始化是使用与Point同名的构造函数Point(int,int)实现的。在1.10.1节介绍使用string的对象str时,使用“string str(real is);”语句。按此推理,可以写出产生Point的对象的语句:Point A(25,56);这就是使用构造函数产生类的实例。不过,现在还没有定

11、义构造函数Point(int,int),所以还不能使用这种方法。但可以看到数据封装的迹象了!第15页/共177页Point类是用户定义的一种类型,Point类所说明的数据成员描述了对象的内部数据结构,对数据成员的访问通过类的成员函数来进行。使用Point在程序中声明变量,具有Point类的类型的变量被称为Point的对象。只有产生类的对象,才能使用这些数据和成员函数。类Point不仅可以声明对象,还可以声明对象的引用和对象的指针,语法与基本数据类型一样。Point A,B;/定义Point类型的对象A和BPoint*p=&A;/定义指向对象A的Point类型的指针Point&R=B;/定义R为

12、Point类型对象B的引用5.1.2 使用类的对象第16页/共177页对象和引用都使用运算符“.”访问对象的成员,指针则使用“-”运算符。例如:A.Setxy(25,88);/为对象A设置初值R.Display();/显示对象B的数据成员 /B.x和B.y之值p-Display();/显示指针p所指对象A的数据 /成员A.x和A.y之值第17页/共177页【例5.3】根据上面对Point类的定义,演示使用Point类的对象。void print(Point a)/使用Point的对象a作为函数参数a.Display();/显示对象a的数据成员的值void main()Point A,B;/声明

13、对象A.Setxy(25,55);/为对象A赋初值B=A;/B的数据成员取A的数据成员之值A.Display();/显示A的数据成员A.Move(-10,20);/移动Aprint(A);/等价于A.Display();print(B);/等价于B.Display()第18页/共177页coutA.Getx()endl;/只能使用A.Getx(),不能使用A.x本例中的print函数使用Point的对象作为参数。C+推荐使用下面的引用的形式:void print(Point&a)/使用对象的引用作为函数参数a.Display();/显示引用对象a的数据成员之值对象A移动之后,对象B仍为原来的值

14、,所以输出如下:25,55/原来的A和B 15,75/新的A 25,55/原来的B 15 /A.Getx()返回对象A的数据成员x的值如果在print函数或主函数里使用如下语句,则产生错误。coutA.x,A.yendl;/错误!第19页/共177页暂不涉及还没有介绍的保护成员,可以归纳出如下规律:类的成员函数可以直接使用自己类的私有成员(数据成员和成员函数);类外面的函数不能直接访问类的私有成员(数据成员和成员函数);类外面的函数只能通过类的对象使用该类的公有成员函数,例如print和main函数;对象A和B的成员函数的代码一样,不同对象的区别是属性的取值。第20页/共177页在程序运行时,

15、通过为对象分配内存来创建对象。在创建对象时,使用类作为样板,故称对象为类的实例。从表5.1中可以看出,A和B两个对象占据内存中的不同区域。A的数据是A.x和A.y,而B的数据是B.x和B.y,它们各有表现自己的属性数据,但用来操作数据的代码均是一样的,例如A.Getx()和B.Getx()的代码一样。为节省内存,在建立对象时,只分配用于保存数据的内存,代码为每个对象共享。类中定义的代码放在计算机内存的一个公用区中,供该类的所有对象共享。这只是C+编译器实现对象的一种方法,作为程序员仍要将对象理解为是由数据和代码组成的。正是由于类拥有这两类成员,才使得数据封装等思想得以实现。第21页/共177页

16、表5.1 类Point的两个实例A和B对象A对象B对象成员对象成员分配内存内容对象成员对象成员分配内存内容A.xA.x的值B.xB.x的值A.yA.y的值B.yB.y的值A.Setxy(int,int)Setxy函数代码B.Setxy(int,int);Setxy函数代码A.Move(int,int)Move函数代码B.Move(int,int);Move函数代码A.Display()Display函数代码B.Display()Display函数代码A.Getx()Getx函数代码B.Getx()Getx函数代码A.Gety()Gety函数代码B.Gety()Gety函数代码第22页/共177

17、页【例5.4】演示使用内联函数定义Point类及使用Point类指针和引用的完整例子。#include/包含头文件using namespace std;/声明命名空间class Point/使用内联函数定义类Pointprivate:/声明为私有访问权限 int x,y;/私有数据成员public:/声明为公有访问权限 void Setxy(int a,int b)/无返回值的内联公有成员函数 x=a;y=b;void Move(int a,int b)/无返回值的内联公有成员函数 x=x+a;y=y+b;void Display()/无返回值的内联公有成员函数 coutx,yDisplay

18、();void print(Point&a)/类引用作为print函数的参数定义重载函数a.Display();void main()/主函数 Point A,B,*p;/声明对象和指针 Point&RA=A;/声明对象RA引用对象A A.Setxy(25,55);/使用成员函数为对象A赋值第24页/共177页 B=A;/使用赋值运算符为对象B赋值 p=&B;/类Point的指针指向对象B p-Setxy(112,115);/使用指针调用函数Setxy重新设置B的值 print(p);/传递指针显示对象B的属性 p-Display();/再次显示对象B的属性 RA.Move(-80,23);/

19、引用对象RA调用Move函数 print(A);/验证RA和A同步变化 print(&A);/直接传递A的地址作为指针参数程序运行结果如下:112,115112,115-55,78-55,78第25页/共177页面向对象的程序设计,是通过为数据和代码建立分块的内存区域,以便提供对程序进行模块化的一种程序设计方法,这些模块可以被用做样板,在需要时再建立其副本。根据这个定义,对象是计算机内存中的一块区域,通过将内存分块,每个模块(即对象)在功能上保持相对独立。另外,定义也表明如下问题。5.1.3 数据封装第26页/共177页 这些内存块中不但存储数据,而且也存储代码,这对保证对象受保护很重要。只有

20、对象中的代码才可以访问存储于这个对象中的数据,这清楚地限定了对象所具有的功能(即一个对象在一个软件系统中所能起到的作用),并使对象保护它自己不受未知的外部事件的影响,从而使自己的数据和功能不会因此遭到破坏。这些内存块的结构可被用做样板产生对象的更多副本。例如,一旦定义了一个窗口对象,只要内存允许,就可以建立许多这样的对象。在面向对象的程序中,对象的行为只有向对象发送消息才能引用,例如通过Display发出显示消息,所以说面向对象是消息处理机制。第27页/共177页对象之间只能通过成员函数调用实现相互通信。这样,对象之间的相互作用方式是仔细控制的,一个对象外部的代码就没有机会通过直接修改对象的内

21、存区域妨碍对象发挥其功能。当对象的一个函数被调用时,对象执行其内部的代码来响应这个调用,这就使对象呈现出一定的行为,即表现出该对象的功能。对象被视为能做出动作的实体,动作在对象相互作用时被激励,换句话说,对象就像在宿主计算机上拥有数据和代码,并能相互通信的具有特定功能的一台较小的计算机。第28页/共177页对象的这一特点导致了模拟现实世界的一种新型方法:面向对象就是将世界看成是由一组彼此相关并能相互通信的实体即对象组成的。程序中的对象映射现实世界中的对象。C+对其对象的数据成员和成员函数的访问是通过访问控制权限来限制的。可以试验一下,将【例5.1】中对数据成员x和y的声明改为public,就可

22、以在主函数中直接使用语句“couty)?x:y;void MaxSet(int x1,int x2,int x3=0,int x4=0)/使用两个默认参数a=x1;b=x2;c=x3;d=x4;int MaxMaxi()/求自己类中4个数的最大值 int x=Maxi(a,b);/x和y为Maxi()函数的局部整数对象 int y=Maxi(c,d);return Maxi(x,y);第33页/共177页/主程序#includeusing namespace std;void main()A0.Set(12,45,76,89);/为数组对象A0置初值A1.Set(12,45,76);/为数组对

23、象A1置初值A2.Set(12,45);/为数组对象A2置初值for(int i=0;i3;i+)/输出对象求值结果 coutAi.Maxi()x=a;this-y=b;这时,成员函数的this指针指向对象A。成员中对x和y的引用表示是引用对象A的成员x和y。对于任何访问该成员函数的类的对象,C+编译器都认为是访问this指针所指向的对象中的成员。由于不同的对象调用成员函数Setxy(int,int)时,this指针指向不同的对象,因此成员函数Setxy(int,int)可以为不同对象的x和y置初值。第37页/共177页使用this指针,保证了每个对象可以拥有自己的数据成员,但处理这些数据成员

24、的代码可以被所有的对象共享。由此可见,C+规定,当一个成员函数被调用时,系统自动向它传递一个隐含的参数,该参数是一个指向接受该函数调用的对象的指针,从而使成员函数知道该对哪个对象进行操作。在程序中,可以使用关键字this来引用该指针。this指针是C+实现封装的一种机制,它将对象和该对象调用的成员函数连接在一起,在外部来,每一个对象都拥有自己的成员函数。如果在定义Setxy函数时使用this指针,也不要给出隐含参数,而写成如下形式:第38页/共177页 void PointSetxy(int a,int b)this-x=a;this-y=b;除非特殊需要,一般情况下都省略符号“this-”,

25、而让系统进行默认设置。第39页/共177页因为类本身就是一种新的数据类型,所以一个类可以作为另一个类的成员。假设有A和B两个类,可以通过在B类里定义A的对象作为B的数据成员,或者定义一个返回类型为A的函数作为B的成员函数。假设定义了坐标点的类Point,矩形类Rectangle的属性需要一个坐标点及长和宽。一个Point的对象恰好可以作为矩形的顶点坐标,即Rectangle可以使用Point的一个对象作为数据成员。在Rectangle类中,用Point类定义一个返回Point类型指针的函数作为Rectangle的成员函数。5.1.6 一个类的对象作为另一个类的成员第40页/共177页【例5.6

26、】使用对象成员的例子。#includeusing namespace std;class Point/定义点类int x,y;public:void Set(int a,int b)x=a;y=b;int Getx()return x;int Gety()return y;class Rectangle/在矩形类里使用Point类的成员 Point Loc;/定义一个Point类的对象作为顶点 int H,W;/H为高,W为宽第41页/共177页 public:void Set(int x,int y,int h,int w);Point*GetLoc();/用Point类定义返回指针的函数

27、int GetHeight()return H;int GetWidth()return W;void Rectangle Set(int x,int y,int h,int w)Loc.Set(x,y);/初始化坐标顶点 H=h;W=w;Point*Rectangle GetLoc()/返回类型Point*,作 /为Rectangle的成员函数 return&Loc;/返回对象Loc的地址第42页/共177页 void main()Rectangle rect;rect.Set(10,2,25,20);coutrect.GetHeight(),rect.GetWidth(),;Point*p

28、=rect.GetLoc();/定义Point类的指针 /对象p并初始化coutGetx(),Gety()Display();注意:不能取私有数据成员的地址,也不能取成员函数的地址。有关指向成员的指针在后面的章节中讨论。指向对象指针的算术运算规则与第4章介绍的一样,不再赘述。第46页/共177页 对象可以用做函数参数,这时参数传递采用传值方式,即在被调用函数中对形参所作的改变不影响调用函数中作为实参的对象。如果要传址,可以采用对象的引用或指针作为参数。但是,如果参数对象被修改,相应的实参对象也将被修改。C+推荐使用引用作为参数传递。为了避免被调用函数修改原来对象的数据成员,可以使用第4章介绍的

29、const修饰符。对象作为函数参数时,可以使用对象值、对象引用和对象地址(指针)。以介绍过的print函数为例,它们的参数传递形式为:void print(Point a)a.Display;void print(Point&a)a.Display;void print(Point*p)p-Display;第47页/共177页它们的原型分别为print(Point),print(Point&)和print(Point*)。对于对象A而言,print(&A)调用的是原型为print(Point*)的函数形式。注意:参数为对象和引用时,编译系统无法区别,重载print只能选择其中的一种。一个对象可

30、以用做另一个对象的成员。例如在【例5.6】中,定义Point类的对象Loc。第48页/共177页1.使用类的权限假设声明了Point类,为了简单具体,讨论数据成员为私有,成员函数为公有的情况。这时会有函数(主函数和一般函数)、对象和类本身的成员函数需要使用Point类的成员。类本身的成员函数可以使用类的所有成员(私有和公有成员)。类的对象只能访问公有成员函数,例如输出x只能使用A.Getx(),不能使用A.x。5.2.2 类的性质第49页/共177页 其他函数不能使用类的私有成员,也不能使用公有成员函数,它们只能通过类的对象使用公有成员函数。一个类可以使用另外一个类的对象,这个类也只能通过那个

31、类的对象使用该类的成员函数,通过成员函数使用数据成员,例如Loc.Set(x,y)。2.不完全的类声明类不是内存中的物理实体,只有当使用类产生对象时,才进行内存分配,这种对象建立的过程称为实例化。实例化一词在面向对象程序设计范围内使用广泛,用以表示产生类的实例或物理实体的动作。在某些语言中,对象就被称作实例。虽然在C+中并不是这样,但实例化仍然被广泛使用。第50页/共177页应当注意的是:类必须在其成员被使用之前进行声明。有时也需要将类作为一个整体来使用,而不存取其成员。声明指针就是这种情况,例如:class MembersOnly;/不完全的类声明 MembersOnly*club;/定义一

32、个全局变量类指针 void main()./函数体/主函数 class MembersOnly./类体;/完全定义该类第一条语句称为不完全类声明,它用于在类没有完全定义之前就引用该类的情况,例如引用另一文件中定义的类。由于类标识符MembersOnly通过不完全类声明进入了作用域,所以就可以声明全局变量指针club。编译器执行到该指针的声明处时,只了解指针所指类型是一个叫MembersOnly的类,而不了解其他任何情况。第51页/共177页不完全声明的类不能实例化,企图实例化会产生编译出错信息;不完全声明仅用于类和结构,企图存取没有完全声明的类成员,也会引起编译出错信息。另外,将数据成员的声明

33、放在最后,有利于先理解类的界面。如果类体内有内联函数,这就像是在声明数据成员之前已经用到它们,确实是“使用在前,说明在后”。这在类中是允许的,因为C+编译器先扫描类的成员说明,最后才处理内联成员函数的代码体。第52页/共177页3.空类尽管类的目的是封装代码和数据,它也可以不包括任何声明。例如:class Empty;这种类没有任何行为,但可以产生空类对象。void main()Empty object;为什么要产生空类呢?在开发大的项目时,需要在一些类还没有完全定义或实现时进行先期测试。这常称为“插头”,用来保证代码能正确地被编译,从而允许测试其中的一部分。不过这是早期的做法,对强类型检查的

34、编译器,会给出一个警告。可给空类增加一个无参数构造函数(见构造函数一节)以消除警告。例如:class Empty public:Empty();第53页/共177页4.类作用域声明类时所使用的一对花括号形成所谓的类作用域。在类作用域中声明的标识符只在类中可见。例如:class example int num;int i=num;/错,num在此不可见 int num;/正确,num与类中说明的数据 /成员num具有不同的作用域即使该成员函数的实现是在类定义之外给出的,类作用域也包含了类中成员函数的作用域。因此,当成员函数内使用一个标识符时,编译器先在包含类作用域中寻找,例如下面的例子:第54页

35、/共177页class MyClass int number;public:void set(int);int number;/这个number不属于类MyClassvoid MyClass set(int i)number=i;/使用类MyClass中的 /标识符number 类中的一个成员名可以使用类名和作用域运算符来显式指定,这称为成员名限定。例如:void MyClass set(int i)第55页/共177页 MyClass number=i;/显式指定访问MyClass /类中的标识符number 在程序中,分析对象的生存期与过去分析变量的生存期的方法一样,由对象说明来决定。类中

36、各数据成员的生存期由对象的生存期决定,当对象存在时,它们存在;对象消失时,它们也消失。成员函数具有外部连接属性。在程序中使用成员选择运算符(“.”或“-”)访问一个对象的成员时,其后的名字是引用该对象所在类中声明的成员名。例如:Point A;/声明类对象,假设类内有成员函数Getx()int Getx()return 5;/类外面声明的函数第56页/共177页 int x=Getx();/访问函数Getx(),返回5 int y=A.Getx();/访问类Point中声明的 /成员函数Getx()在类中说明的枚举成员也使用成员选择运算符存取,枚举成员名隐藏在类作用域中。这些枚举成员不属于任何

37、对象,为该类所有的对象共享。因此,对这些枚举成员使用成员名限定方法进行存取比较恰当。但是,在一个友元函数访问类中的枚举成员时,有时必须使用成员选择运算符。友元函数在后面的章节讨论。除非编译器在处理类声明时遇到了标识其结束的右花括号,否则这个声明仍然是引用性声明。第57页/共177页引用性声明所声明的类名不能用来建立对象,只能用来声明指针或引用,或用在函数声明中。例如:class MyClass MyClass member;/错MyClass*p;/正确;class YourClass private:MyClass d;/正确;第58页/共177页在上例中,当在类MyClass中声明成员me

38、mber时,类名MyClass仅作了引用性声明,因而这个语句是错误的。由此可见,C+为类中声明的标识符引入了新的类作用域概念。这些标识符只有与类对象连用时才进入作用域。在可能出现二义性的情况下,必须使用作用域限定符“”。作用域限定符“”的特点将在后继章节中讨论。第59页/共177页在C+中,封装也可以由struct和union等关键字提供,它们也能将数据和函数组合成一个类,所以也可以使用结构和联合来定义类。结构是类的一种特例,其中成员在默认情况下是公有的。尽管C+允许在结构中定义成员函数,但它本身更适合处理数据,例如像职工记录和日期这样的数据结构。联合以关键字union来声明,成员默认为公有并

39、且在某个给定时间,只出现一个成员。联合使若干数据成员使用同一地址。可以直接在类中使用无名联合,例如:5.3 结构与联合第60页/共177页 Class CU union int ivalue;float fvalue;/在关键字union后面没有给出联合名,这是一个无名联合声明,它声明ivalue和fvalue共享同一块内存,无名联合中声明的数据项名字可以被直接存取。第61页/共177页无名联合不仅可以用于声明类中的数据成员,而且可用于程序中需要共享数据项的其他地方,例如在函数作用域中,或在文件作用域中。无名联合不能有成员函数,因为无名联合中成员的作用域在联合之外。目前介绍的类都是与其他无关的

40、单一的类。如果能通过这些已有的类来建立新类,则新建立的类叫做“派生类”,而原来的类称为“基类”。这些将在第6章详细介绍,目前只是强调联合既不能用做任何类的基类,也不能从任何类中派生出联合,因为联合在特定的时刻只有一个数据成员处于“激活”状态。此外,在联合中也不能说明虚函数。第62页/共177页建立一个对象时,对象的状态(数据成员的取值)是不确定的。为了使对象的状态确定,必须对其进行正确的初始化。C+有称为构造函数的特殊成员函数,它可自动进行对象的初始化。初始化和赋值是不同的操作,当C+语言预先定义的初始化和赋值操作不能满足程序的要求时,程序员可以定义自己的初始化和赋值操作。5.4 构造函数第6

41、3页/共177页由于初始化的重要性,调用构造函数并不是由用户来完成的;是否调用这个函数也不是可选的,而是由编译器来调用,所以编译器必须总能知道调用哪个函数。最容易也最符合逻辑的方法是指定这函数的名称与类名一样。这个函数没有返回值则是基于下面的考虑:如果它有返回值,编译器就必须知道如何处理返回值,这样就会大大增加编译器的工作,也降低了效率。但如果采取只能由用户自己来显式调用构造函数方式的话,就又容易破坏安全性。5.4.1 定义构造函数第64页/共177页因此,让构造函数的名字和类名同名,在定义构造函数时不能指定返回类型,即使是void 类型也不可以。【例5.7】的程序说明构造函数的定义和执行过程

42、。【例5.7】构造函数的定义和执行过程实例程序。class Test private:int num;public:Test();/声明不带参数的构造函数Test(int);/声明带1个参数的构造函数;第65页/共177页#includeusing namespace std;Test Test():num(0)/定义不带参数的构造函数coutInitializing defaultendl;TestTest(int n):num(n)/定义带1个参数的构造函数coutInitializing nendl;void main()Test x;/使用不带参数的构造函数产生对象x Test y(1

43、5);/使用带参数的构造函数产生对象y Test array2=5,7;/使用带参数的构造函数 /产生对象数组array第66页/共177页程序中和Test类同名的两个成员函数是构造函数,一个不带参数,一个带有1个参数。“:num(n)”完成“num=n”的功能,这里称初始化列表。它与下面的方法等价:TestTest(int n)coutInitializing nendl;num=n;由此可见,类可以有多个构造函数,在类体里的声明形式如下:类名(形参1,形参2,形参n);/可以没有形参类的构造函数可以在类体内(内联函数)声明时定义,也可以在类体外定义。可以使用初始化列表或者在构造函数的函数体

44、内定义。第67页/共177页假设数据成员为x1,x2,xn,则有如下两种形式:类名类名(形参1,形参2,形参n):x1(形参1),x2(形参2),xn(形参n)类名类名(形参1,形参2,形参n)x1=形参1;x2=形参2;xn=形参n;构造函数的参数是无顺序排列,只要保证相互的对应顺序即可。可以使用默认参数或重载。第68页/共177页在程序中说明一个对象时,程序自动调用构造函数来初始化该对象。当执行到语句 Test x;时,程序为对象x分配内存,然后调用不带参数的构造函数来初始化这段内存,将x的数据成员num 初始化为零,输出“Initailizing default”。当程序执行到语句 Te

45、st y(15);时,程序为对象y分配内存,然后调用带有参数的构造函数来初始化这段内存,将y的数据成员 num 初始化为15,输出“Initializing 15”。第69页/共177页当程序执行到语句 Test array 2=5,7时,程序为对象数组array2分配内存,然后调用带参数的构造函数来初始化这段内存。它必须为每一个数组元素调用一次构造函数,先将array0的数据成员 num 初始化为5,输出“Initializing 5”;再将array1的数据成员 num 初始化为7,输出“Initializing 7”。当声明一个外部对象时,外部对象只是引用在其他地方声明的对象,程序并不为

46、外部对象说明调用构造函数。如果是全局对象或静态对象,在 main 函数执行之前要调用它们的构造函数。第70页/共177页使用下面的主程序演示全局对象的情况:/使用前面定义的类 Test global(5);void main()coutEntering main and exiting main endl;程序在进入main函数之前先构造全局对象,输出 Initializing 5进入主函数之后输出为:Entering main and exiting main。第71页/共177页运算符new用于建立生存期可控的对象,new返回这个对象的指针。由于类名被视为一个类型名,因此,使用new建立动

47、态对象的语法和建立动态变量的情况类似,其不同点是new和构造函数一同起作用。下面程序使用前面的类定义,说明如何使用new运算符及其工作过程。5.4.2 构造函数和运算符new第72页/共177页/使用前面的类定义 void main()Test*ptr1=new Test;Test*ptr2=new Test(5);delete ptr1;delete ptr2;运行这个程序,输出结果是:Initializing default Initializing 5第73页/共177页当使用 new 建立一个动态对象时,new 首先分配足以保存Test 类的一个对象所需要的内存,然后自动调用构造函数来

48、初始化这块内存,再返回这个动态对象的地址。和说明对象时的情况一样,对于表达式 new Test;new 调用不带参数的构造函数,输出 Initializing default而对于表达式 new Test(5);new 调用了带参数的构造函数,输出 Initializing 5使用new建立的动态对象在不用时必须用delete删除,以便释放所占空间。第74页/共177页不带参数的构造函数又称做默认构造函数,原因是当没有为一个类定义任何构造函数的情况下,C+编译器总要自动建立一个不带参数的构造函数。例如,如果在上面的例子中没有为Test 类定义任何构造函数,则C+编译器要为它产生一个默认构造函数

49、,这个默认构造函数的形式如下:TestTest()即它的函数体是空的。因此,当建立 Test 类的一个对象时,对象的状态是不确定的,即没有被初始化。5.4.3 默认构造函数和默认参数第75页/共177页一旦程序定义了自己的构造函数,系统就不再提供这个默认构造函数。如果程序员没有再定义一个无参数的构造函数,但又声明了一个没有初始化的对象(如“Test A;”),则因系统已经不再提供默认构造函数,编译就会出错。同理,如果只声明对象数组(即不提供初值),因为每个元素对象均需要调用一次默认构造函数来为自己初始化,所以必须要求有一个默认构造函数。如果程序中定义了有参数的构造函数,千万别忘记为它定义一个无

50、参数的构造函数。需要说明的是“默认”也称为“缺省”,本书统一使用“默认”。解决的另一种方法是将相应的构造函数全部使用默认参数设计。第76页/共177页【例5.8】设计构造函数的默认参数。#includeusing namespace std;class Test private:int x,y;public:Test(int=0,int=0);/声明两个参数均为默认参数;TestTest(int a,int b):x(a),y(b)/定义构造函数 coutInitializing x,yendl;void main()Test A;Test B(15,55);Test array2;第77页/

展开阅读全文
相关资源
相关搜索

当前位置:首页 > 应用文书 > PPT文档

本站为文档C TO C交易模式,本站只提供存储空间、用户上传的文档直接被用户下载,本站只是中间服务平台,本站所有文档下载所得的收益归上传人(含作者)所有。本站仅对用户上传内容的表现方式做保护处理,对上载内容本身不做任何修改或编辑。若文档所含内容侵犯了您的版权或隐私,请立即通知淘文阁网,我们立即给予删除!客服QQ:136780468 微信:18945177775 电话:18904686070

工信部备案号:黑ICP备15003705号© 2020-2023 www.taowenge.com 淘文阁