谭浩强版《C++程序设计》第9章n.ppt
《谭浩强版《C++程序设计》第9章n.ppt》由会员分享,可在线阅读,更多相关《谭浩强版《C++程序设计》第9章n.ppt(89页珍藏版)》请在三一办公上搜索。
1、第9章 关于类和对象的进一步讨论,9.1 构造函数9.2 析构函数9.3 调用构造函数和析构函数的顺序9.4 对象数组9.5 对象指针9.6 共用数据的保护9.7 对象的动态建立和释放9.8 对象的赋值和复制9.9 静态成员9.10 友元9.11 类模板,在建立一个对象时,作某些初始化的工作如对数据成员赋初值。即在创建对象(分配内存时)进行数据成员的初始化,因为对象是实实在在的对象,不能无具体属性值。注意:类的数据成员是不能在声明类时初始化的。,9.1 构造函数作用:创建对象(分配内存时)时进行数据成员的初始化9.1.1 对象的初始化,如果一个类中所有的成员都是公用的,则可以在定义对象时对数据
2、成员进行初始化。如class Timepublic:/声明为公用成员 hour;minute;sec;Time t1=14,56,30;/将t1初始化为14:56:30但是,一般数据成员是私有的,或者类中有private或protected的成员,就不能用这种方法初始化。如何实现?,C+提供了构造函数(constructor)来处理对象的初始化。构造函数是特殊的成员函数,与其他成员函数不同,不需要用户来调用它,而是在建立对象时自动执行。构造函数的名字必须与类名同名,而不能由用户任意命名,以便编译系统能识别它并把它作为构造函数处理。它不具有任何类型,不返回任何值。构造函数的功能是由用户定义的,用
3、户根据初始化的要求设计函数体和函数参数。,9.1.2 构造函数的作用,例9.1 在例8.3基础上定义构造成员函数。#include using namespace std;class Timepublic:Time()/定义构造成员函数,函数名与类名相同hour=0;/利用构造函数对对象中的数据成员赋初值minute=0;sec=0;void set_time();/函数声明void show_time();/函数声明private:int hour;/私有数据成员int minute;int sec;,void Timeset_time()/定义成员函数,向数据成员赋值cinhour;cin
4、minute;cinsec;void Timeshow_time()/定义成员函数,输出数据成员的值 couthour:minute:secendl;int main()Time t1;/建立对象t1,同时调用构造函数t1.Time()t1.set_time();/对t1的数据成员赋值t1.show_time();/显示t1的数据成员的值 Time t2;/建立对象t2,同时调用构造函数t2.Time()t2.show_time();/显示t2的数据成员的值return 0;,程序运行的情况为:10 25 54(从键盘输入新值赋给t1的数据成员)10:25:54(输出t1的时、分、秒值)0:0
5、:0(输出t2的时、分、秒值)也可以在类外定义构造函数:TimeTime()/要加上类名Time和域限定 符“”hour=0;minute=0;sec=0;,有关构造函数的使用,有以下说明:(1)在类对象进入其作用域时调用构造函数。(2)构造函数没有返回值,因此也不需要在定义构造函数时声明类型,这是它和一般函数的一个重要的不同之点。(3)构造函数不需用户调用,也不能被用户调用。(4)如果用户自己没有定义构造函数,则C+系统会自动生成一个构造函数,只是这个构造函数的函数体是空的,也没有参数,不执行初始化操作。,不带参数构造函数,这种方式使该类的每一个对象都得到同一组初值。带参数的构造函数,用户希
6、望对不同的对象赋予不同的初值。构造函数首部的一般格式:构造函数名(类型 1 形参1,类型2 形参2,)实参是在定义对象时给出的。定义对象的一般格式为:类名 对象名(实参1,实参2,);,9.1.3 带参数的构造函数,例9.2 有两个长方柱,其长、宽、高分别为:(1)12,20,25;(2)10,14,20。求它们的体积。编一个基于对象的程序,在类中用带参数的构造函数。#include using namespace std;class Boxpublic:Box(int,int,int);/声明带参数的构造函数int volume();/声明计算体积的函数 private:int height
7、;int width;int length;BoxBox(int h,int w,int len)/在类外定义带参数的构造函数height=h;width=w;length=len;,int Boxvolume()/定义计算体积的函数return(height*width*length);int main()Box box1(12,25,30);/建立对象box1,并指定box1长、宽、高的值coutThe volume of box1 is box1.volume()endl;Box box2(15,30,21);/建立对象box2,并指定box2长、宽、高的值coutThe volume
8、of box2 is box2.volume()endl;return 0;程序运行结果如下:The volume of box1 is 9000The volume of box2 is 9450注意:带参数的构造函数中的形参,其对应的实参在定义对象时给定。,C+还提供另一种初始化数据成员的方法参数初始化表来实现对数据成员的初始化。这种方法不在函数体内对数据成员初始化,而是在函数首部实现。例如例9.2中定义构造函数可以改用以下形式:BoxBox(int h,int w,int len):height(h),width(w),length(len)这种写法方便、简练,尤其当需要初始化的数据成员
9、较多时更显其优越性。甚至可以直接在类体中(而不是在类外)定义构造函数。,9.1.4 用参数初始化表对数据成员初始化,在一个类中可以定义多个构造函数,以便对类对象提供不同的初始化的方法,供用户选用。这些构造函数具有相同的名字,而参数的个数或参数的类型不相同。这称为构造函数的重载。在第4章第4.6节中所介绍的函数重载的知识也适用于构造函数。通过下面的例子可以了解怎样应用构造函数的重载。,9.1.5 构造函数的重载,例9.3 在例9.2的基础上,定义两个构造函数,其中一个无参数,一个有参数。#include using namespace std;class Boxpublic:Box();/声明一
10、个无参的构造函数Box(int h,int w,int len):height(h),width(w),length(len)/声明一个有参的构造函数,用参数的初始化表对数据成员初始化int volume();private:int height;int width;int length;BoxBox()/定义一个无参的构造函数height=10;width=10;length=10;,int Boxvolume()return(height*width*length);int main()Box box1;/建立对象box1,不指定实参coutThe volume of box1 is bo
11、x1.volume()endl;Box box2(15,30,25);/建立对象box2,指定3个实参coutThe volume of box2 is box2.volume()endl;return 0;在本程序中定义了两个重载的构造函数,其实还可以定义其他重载构造函数。,说明:(1)调用构造函数时不必给出实参的构造函数,称为默认构造函数(default constructor)。显然,无参的构造函数属于默认构造函数。一个类只能有一个默认构造函数。(2)如果在建立对象时选用的是无参构造函数,应注意正确书写定义对象的语句。(3)尽管在一个类中可以包含多个构造函数,但是对于每一个对象来说,建立
12、对象时只执行其中一个构造函数,并非每个构造函数都被执行。,构造函数中参数的值既可以通过实参传递,也可以指定为某些默认值,即如果用户不指定实参值,编译系统就使形参取默认值。例9.4 将例9.3程序中的构造函数改用含默认值的参数,长、宽、高的默认值均为10。在例9.3程序的基础上改写如下:,9.1.6 使用默认参数的构造函数,#include using namespace std;class Boxpublic:Box(int h=10,int w=10,int len=10);/在声明构造函数时指定默认参数int volume();private:int height;int width;in
13、t length;BoxBox(int h,int w,int len)/在定义函数时可以不指定默认参数height=h;width=w;length=len;,int Boxvolume()return(height*width*length);int main()Box box1;/没有给实参 coutThe volume of box1 is box1.volume()endl;Box box2(15);/只给定一个实参coutThe volume of box2 is box2.volume()endl;Box box3(15,30);/只给定2个实参coutThe volume of
14、 box3 is box3.volume()endl;Box box4(15,30,20);/给定3个实参coutThe volume of box4 is box4.volume()endl;return 0;,析构函数(destructor)也是一个特殊的成员函数,它的作用与构造函数相反,它的名字是类名的前面加一个“”符号。在C+中“”是位取反运算符,从这点也可以想到:析构函数是与构造函数作用相反的函数。当对象的生命期结束时,会自动执行析构函数。如果在一个函数中定义了一个对象(它是自动局部对象),当这个函数被调用结束时,对象应该释放,在对象释放前自动执行析构函数。,9.2 析构函数,sta
15、tic局部对象,只在main函数结束或调用exit函数结束程序时,才调用static局部对象的析构函数。全局对象,则在程序的流程离开其作用域时(如main函数结束或调用exit函数)时,调用该全局对象的析构函数。如果用new运算符动态地建立了一个对象,当用delete运算符释放该对象时,先调用该对象的析构函数。析构函数的作用:并不是删除对象,而是在撤销对象占用的内存之前完成一些清理工作,使这部分内存可以被程序分配给新对象使用。程序设计者事先设计好析构函数,只要对象的生命期结束,程序就自动执行析构函数来完成这些工作。,析构函数不返回任何值,没有函数类型,也没有函数参数。因此它不能被重载。一个类可
16、以有多个构造函数,但只能有一个析构函数。析构函数的作用并不仅限于释放资源方面,它还可以输出有关的信息。一般情况下,类的设计者应当在声明类的同时定义析构函数,以指定如何完成“清理”的工作。如果用户没有定义析构函数,C+编译系统会自动生成一个析构函数,但它只是徒有析构函数的名称和形式,实际上什么操作都不进行。想让析构函数完成任何工作,都必须在定义的析构函数中指定。,例9.5 包含构造函数和析构函数的C+程序。#include#includeusing namespace std;class Student/声明Student类public:student(int n,string nam,char
17、 s)/定义构造函数num=n;name=nam;sex=s;coutConstructor called.endl;/输出有关信息Student()/定义析构函数coutDestructor called.endl;/输出有关信息void display()/定义成员函数coutnum:numendl;coutname:nameendl;coutsex:sexendlendl;,private:int num;char name10;char sex;int main()Student stud1(10010,Wang_li,f);/建立对象stud1stud1.display();/输出学
18、生1的数据 Student stud2(10011,Zhang_fun,m);/定义对象stud2stud2.display();/输出学生2的数据return 0;,程序运行结果如下:Constructor called.(执行stud1的构造函数)num:10010(执行stud1的display函数)name:Wang_lisex:fConstructor called.(执行stud2的构造函数)num:10011(执行stud2的display函数)name:Zhang_funsex:mDestructor called.(执行stud2的析构函数)Destructor called
19、.(执行stud1的析构函数),在使用构造函数和析构函数时,需要特别注意对它们的调用时间和调用顺序。在一般情况下,调用析构函数的次序正好与调用构造函数的次序相反:最先被调用的构造函数,其对应的(同一对象中的)析构函数最后被调用,而最后被调用的构造函数,其对应的析构函数最先被调用。如图9.1示意。,9.3 调用构造函数和析构函数的顺序,图9.1,但是,并不是在任何情况下都是按这一原则处理的。对象可以在不同的作用域中定义,可以有不同的存储类别。这些会影响调用构造函数和析构函数的时机。下面归纳一下什么时候调用构造函数和析构函数:(1)在全局范围中定义的对象它的构造函数在文件中的所有函数(包括main
20、函数)执行之前调用。当main函数执行完毕或调用exit函数时(此时程序终止),调用析构函数。,(2)如果定义的是局部自动对象(例如在函数中定义对象),则在建立对象时调用其构造函数。如果函数被多次调用,则在每次建立对象时都要调用构造函数。在函数调用结束、对象释放时先调用析构函数。(3)如果在函数中定义静态(static)局部对象,则只在程序第一次调用此函数建立对象时调用构造函数一次,只在main函数结束或调用exit函数结束程序时,才调用析构函数。,数组也可以由对象组成(对象数组的每一个元素都是同类的对象)。例如一个班有50个学生,每个学生的属性包括姓名、性别、年龄、成绩等。如果为每一个学生建
21、立一个对象,需要分别取50个对象名。用程序处理很不方便。这时可以定义一个“学生类”对象数组,每一个数组元素是一个“学生类”对象。例如 Student stud50;/假设已声明了Student类,定义stud数组,有50个元素,9.4 对象数组,在建立数组时,同样要调用构造函数。如果有50个元素,需要调用50次构造函数。如果构造函数有多个参数,则不能用在定义数组时直接提供所有实参的方法,因为一个数组有多个元素,对每个元素要提供多个实参,如果再考虑到构造函数有默认参数的情况,很容易造成实参与形参的对应关系不清晰,出现歧义性。,如果构造函数有多个参数,在定义对象数组时应当怎样实现初始化呢?在花括号
22、中分别写出构造函数并指定实参。如果构造函数有3个参数,分别代表学号、年龄、成绩。则可以这样定义对象数组:Student Stud3=/定义对象数组Student(1001,18,87),/调用第1个元素的构造函数,为它提供3个实参Student(1002,19,76),/调用第2个元素的构造函数,为它提供3个实参Student(1003,18,72)/调用第3个元素的构造函数,为它提供3个实参;,在建立对象数组时,分别调用构造函数,对每个元素初始化。每一个元素的实参分别用括号包起来,对应构造函数的一组形参,不会混淆。例9.6 对象数组的使用方法。#include using namespace
23、 std;class Boxpublic:Box(int h=10,int w=12,int len=15):height(h),width(w),length(len)/声明有默认参数的构造函数,用参数初始化表对数据成员初始化int volume();private:int height;int width;,int length;int Boxvolume()return(height*width*length);int main()Box a3=/定义对象数组Box(10,12,15),/调用构造函数Box,提供第1个元素的实参Box(15,18,20),/调用构造函数Box,提供第2个
24、元素的实参Box(16,20,26)/调用构造函数Box,提供第3个元素的实参;coutvolume of a0 is a0.volume()endl;coutvolume of a1 is a1.volume()endl;coutvolume of a2 is a2.volume()endl;,运行结果如下:volume of a0 is 1800volume of a1 is 5400volume of a2 is 8320,在建立对象时,编译系统会为每一个对象分配一定的存储空间,以存放其成员。对象空间的起始地址就是对象的指针。可以定义一个指针变量,用来存放对象的指针。如果有一个类:cla
25、ss Timepublic:int hour;int minute;int sec;void get_time();,9.5 对象指针 9.5.1 指向对象的指针,void Timeget_time()couthour pt所指向的对象中的hour成员,即t1.hour(*pt).get_time()调用pt所指向的对象中的get_time函数,即t1.get_timept-get_time()调用pt所指向的对象中的get_time函数,即t1.get_time,对象有地址,存放对象初始地址的指针变量就是指向对象的指针变量。对象中的成员也有地址,存放对象成员地址的指针变量就是指向对象成员的指
- 配套讲稿:
如PPT文件的首页显示word图标,表示该PPT已包含配套word讲稿。双击word图标可打开word文档。
- 特殊限制:
部分文档作品中含有的国旗、国徽等图片,仅作为作品整体效果示例展示,禁止商用。设计者仅对作品中独创性部分享有著作权。
- 关 键 词:
- C+程序设计 谭浩强版 C+ 程序设计
链接地址:https://www.31ppt.com/p-5318329.html