




版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领
文档简介
1、第第9章章 关于类和对象的进一步讨论关于类和对象的进一步讨论 9.1 构造函数构造函数 9.2 析构函数析构函数 9.3 调用构造函数和析构函数的顺序调用构造函数和析构函数的顺序 9.4 对象数组对象数组 9.5 对象指针对象指针 9.6 共用数据的保护共用数据的保护 9.7 对象的动态建立和释放对象的动态建立和释放 9.8 对象的赋值和复制对象的赋值和复制 9.9 静态成员静态成员 9.10 友元友元 9.11 类模板类模板 在建立一个对象时,常常需要作某些初始化的工作,在建立一个对象时,常常需要作某些初始化的工作, 例如对数据成员赋初值。例如对数据成员赋初值。如果一个数据成员未被赋如果一个
2、数据成员未被赋 值,则它的值是不可预知的,因为在系统为它分配值,则它的值是不可预知的,因为在系统为它分配 内存时,保留了这些存储单元的原状,这就成为了内存时,保留了这些存储单元的原状,这就成为了 这些数据成员的初始值。这种状况显然是与人们的这些数据成员的初始值。这种状况显然是与人们的 要求不相符的,对象是一个实体,它反映了客观事要求不相符的,对象是一个实体,它反映了客观事 物的属性物的属性(例如时钟的时、分、秒的值例如时钟的时、分、秒的值),是应该有确,是应该有确 定的值的。定的值的。 注意:注意: 类的数据成员是不能在声明类时初始化的类的数据成员是不能在声明类时初始化的。 9.1 构造函数构
3、造函数 9.1.1 对象的初始化对象的初始化 如果一个类中所有的成员都是公用的,则可以在定如果一个类中所有的成员都是公用的,则可以在定 义对象时对数据成员进行初始化。义对象时对数据成员进行初始化。如如 class Time public: /声明为公用成员声明为公用成员 hour; minute; sec; ; Time t1=14,56,30; /将将t1初始化为初始化为14:56:30 这种情况和结构体变量的初始化是差不多的,在一这种情况和结构体变量的初始化是差不多的,在一 个花括号内顺序列出各公用数据成员的值,两个值个花括号内顺序列出各公用数据成员的值,两个值 之间用逗号分隔。之间用逗号
4、分隔。但是,如果数据成员是私有的,但是,如果数据成员是私有的, 或者类中有或者类中有private或或protected的成员,就不能用这的成员,就不能用这 种方法初始化。种方法初始化。 在第在第8章的几个例子中,是用成员函数来对对象中的章的几个例子中,是用成员函数来对对象中的 数据成员赋初值的数据成员赋初值的(例如例例如例8.3中的中的set_time函数函数)。从。从 例例8.3中可以看到,用户在主函数中调用中可以看到,用户在主函数中调用set_time函数函数 来为数据成员赋值。来为数据成员赋值。如果对一个类定义了多个对象,如果对一个类定义了多个对象, 而且类中的数据成员比较多,那么,程
5、序就显得非而且类中的数据成员比较多,那么,程序就显得非 常臃肿烦琐。常臃肿烦琐。 为了解决这个问题,为了解决这个问题,C+提供了提供了构造函数构造函数 (constructor)来来处理对象的初始化处理对象的初始化。构造函数是一种。构造函数是一种 特殊的成员函数,与其他成员函数不同,不需要用特殊的成员函数,与其他成员函数不同,不需要用 户来调用它,而是在户来调用它,而是在建立对象时自动执行建立对象时自动执行。构造函构造函 数的名字必须与类名同名数的名字必须与类名同名,而不能由用户任意命名,而不能由用户任意命名, 以便编译系统能识别它并把它作为构造函数处理。以便编译系统能识别它并把它作为构造函数
6、处理。 它不具有任何类型,不返回任何值。它不具有任何类型,不返回任何值。构造函数的功构造函数的功 能是由用户定义的,用户根据初始化的要求设计函能是由用户定义的,用户根据初始化的要求设计函 数体和函数参数。数体和函数参数。 9.1.2 构造函数的作用构造函数的作用 例例9.1 在例在例8.3基础上定义构造成员函数。基础上定义构造成员函数。 #include using namespace std; class Time public: Time( ) /定义构造成员函数,函数名与类名相同定义构造成员函数,函数名与类名相同 hour=0; /利用构造函数对对象中的数据成员赋初值利用构造函数对对象中
7、的数据成员赋初值 minute=0; sec=0; void set_time( ); /函数声明函数声明 void show_time( ); /函数声明函数声明 private: int hour; /私有数据成员私有数据成员 int minute; int sec; ; void Time set_time( ) /定义成员函数,向数据成员赋值定义成员函数,向数据成员赋值 cinhour; cinminute; cinsec; void Time show_time( ) /定义成员函数,输出数据成员的值定义成员函数,输出数据成员的值 couthour:minute:secendl; i
8、nt 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
9、 (输出输出t1的时、分、秒值的时、分、秒值) 0:0:0 (输出输出t2的时、分、秒值的时、分、秒值) 上面是在类内定义构造函数的,也可以只在类内对上面是在类内定义构造函数的,也可以只在类内对 构造函数进行声明而在类外定义构造函数构造函数进行声明而在类外定义构造函数。将程序。将程序 中的第中的第47行改为下面一行:行改为下面一行: Time( ); /对构造函数进行声明对构造函数进行声明 在类外定义构造函数:在类外定义构造函数: Time Time( ) /在类外定义构造成员函数,要加上类名在类外定义构造成员函数,要加上类名Time和域限定符和域限定符 “ ” hour=0; minute=
10、0; sec=0; 有关构造函数的使用,有以下说明:有关构造函数的使用,有以下说明: (1) 在类对象进入其作用域时调用构造函数在类对象进入其作用域时调用构造函数。 (2) 构造函数没有返回值构造函数没有返回值,因此也不需要在定义构造,因此也不需要在定义构造 函数时声明类型,这是它和一般函数的一个重要的函数时声明类型,这是它和一般函数的一个重要的 不同之点。不同之点。 (3) 构造函数不需用户调用,也不能被用户调用构造函数不需用户调用,也不能被用户调用。 (4) 在构造函数的函数体中不仅可以对数据成员赋初在构造函数的函数体中不仅可以对数据成员赋初 值,而且可以包含其他语句。值,而且可以包含其他
11、语句。但是一般不提倡在构但是一般不提倡在构 造函数中加入与初始化无关的内容,以保持程序的造函数中加入与初始化无关的内容,以保持程序的 清晰。清晰。 (5) 如果用户自己没有定义构造函数,则如果用户自己没有定义构造函数,则C+系统会系统会 自动生成一个构造函数,只是这个构造函数的函数自动生成一个构造函数,只是这个构造函数的函数 体是空的,也没有参数,不执行初始化操作体是空的,也没有参数,不执行初始化操作。 在例在例9.1中构造函数不带参数,在函数体中对数据成中构造函数不带参数,在函数体中对数据成 员赋初值。这种方式使该类的每一个对象都得到同员赋初值。这种方式使该类的每一个对象都得到同 一组初值一
12、组初值(例如例例如例9.1中各数据成员的初值均为中各数据成员的初值均为0)。但但 是有时用户希望对是有时用户希望对不同的对象赋予不同的初值不同的对象赋予不同的初值。 可以可以采用带参数的构造函数采用带参数的构造函数,在调用不同对象的构在调用不同对象的构 造函数时,从外面将不同的数据传递给构造函数,造函数时,从外面将不同的数据传递给构造函数, 以实现不同的初始化。以实现不同的初始化。构造函数首部的一般格式为构造函数首部的一般格式为 构造函数名构造函数名(类型类型 1 形参形参1,类型,类型2 形参形参2,) 前面已说明:前面已说明: 用户是不能调用构造函数的,因此无用户是不能调用构造函数的,因此
13、无 法采用常规的调用函数的方法给出实参法采用常规的调用函数的方法给出实参。实参是在实参是在 定义对象时给出的。定义对象时给出的。定义对象的一般格式为定义对象的一般格式为 类名类名 对象名对象名(实参实参1,实参,实参2,); 9.1.3 带参数的构造函数带参数的构造函数 例例9.2 有两个长方柱,其长、宽、高分别为:有两个长方柱,其长、宽、高分别为: (1)12,20,25;(2)10,14,20。求它们的体积。编一个基。求它们的体积。编一个基 于对象的程序,在类中用带参数的构造函数。于对象的程序,在类中用带参数的构造函数。 #include using namespace std; clas
14、s Box public: Box(int,int,int); /声明带参数的构造函数声明带参数的构造函数 int volume( ); /声明计算体积的函数声明计算体积的函数 private: int height; int width; int length; ; Box Box(int h,int w,int len) /在类外定义带参数的构造函数在类外定义带参数的构造函数 height=h; width=w; length=len; int Box volume( ) /定义计算体积的函数定义计算体积的函数 return(height*width*length); int main(
15、) Box box1(12,25,30); /建立对象建立对象box1,并指定并指定box1长、宽、高的值长、宽、高的值 coutThe volume of box1 is box1.volume( )endl; Box box2(15,30,21); /建立对象建立对象box2,并指定并指定box2长、宽、高的值长、宽、高的值 coutThe volume of box2 is box2.volume( )endl; return 0; 程序运行结果如下:程序运行结果如下: The volume of box1 is 9000 The volume of box2 is 9450 可以知道:
16、可以知道: (1) 带参数的构造函数中的形参,其对应的实参在带参数的构造函数中的形参,其对应的实参在 定义对象时给定定义对象时给定。 (2) 用这种方法可以方便地实现对不同的对象进行用这种方法可以方便地实现对不同的对象进行 不同的初始化不同的初始化。 在在9.1.3节中介绍的是在构造函数的函数体内通过赋节中介绍的是在构造函数的函数体内通过赋 值语句对数据成员实现初始化。值语句对数据成员实现初始化。C+还提供另一种初还提供另一种初 始化数据成员的方法始化数据成员的方法参数初始化表参数初始化表来实现对数来实现对数 据成员的初始化。这种方法不在函数体内对数据成据成员的初始化。这种方法不在函数体内对数
17、据成 员初始化,而是在函数首部实现。例如例员初始化,而是在函数首部实现。例如例9.2中定义中定义 构造函数可以改用以下形式:构造函数可以改用以下形式: Box Box(int h,int w,int len):height(h),width(w),length(len) 这种写法方便、简练,尤其当需要初始化的数据成这种写法方便、简练,尤其当需要初始化的数据成 员较多时更显其优越性。甚至可以直接在类体中员较多时更显其优越性。甚至可以直接在类体中(而而 不是在类外不是在类外)定义构造函数。定义构造函数。 9.1.4 用参数初始化表对数据成员初始化用参数初始化表对数据成员初始化 在一个类中可以在一个
18、类中可以定义多个构造函数定义多个构造函数,以便对类对象,以便对类对象 提供不同的初始化的方法提供不同的初始化的方法,供用户选用。这些构造,供用户选用。这些构造 函数具有函数具有相同的名字相同的名字,而,而参数的个数或参数的类型参数的个数或参数的类型 不相同不相同。这称为构造函数的重载。这称为构造函数的重载。在第在第4章第章第4.6节中节中 所介绍的函数重载的知识也适用于构造函数。所介绍的函数重载的知识也适用于构造函数。 通过下面的例子可以了解怎样应用构造函数的重载。通过下面的例子可以了解怎样应用构造函数的重载。 9.1.5 构造函数的重载构造函数的重载 例例9.3 在例在例9.2的基础上,定义
19、两个构造函数,其中一的基础上,定义两个构造函数,其中一 个无参数,一个有参数。个无参数,一个有参数。 #include using namespace std; class Box public: Box( ); /声明一个无参的构造函数声明一个无参的构造函数 Box(int h,int w,int len):height(h),width(w),length(len) /声明一个有参的构造函数,用参数的初始化表对数据成员初始化声明一个有参的构造函数,用参数的初始化表对数据成员初始化 int volume( ); private: int height; int width; int leng
20、th; ; Box Box( ) /定义一个无参的构造函数定义一个无参的构造函数 height=10; width=10; length=10; int Box volume( ) return(height*width*length); int main( ) Box box1; /建立对象建立对象box1,不指定实参不指定实参 coutThe volume of box1 is box1.volume( )endl; Box box2(15,30,25); /建立对象建立对象box2,指定指定3个实参个实参 coutThe volume of box2 is box2.volume( )e
21、ndl; return 0; 在本程序中定义了两个重载的构造函数,其实还可在本程序中定义了两个重载的构造函数,其实还可 以定义其他重载构造函数,其原型声明可以为以定义其他重载构造函数,其原型声明可以为 Box Box(int h); /有有1个参数的构造函数个参数的构造函数 Box Box(int h,int w); /有两个参数的构造函数有两个参数的构造函数 在建立对象时分别给定在建立对象时分别给定1个参数和个参数和2个参数。个参数。 说明:说明: (1) 调用构造函数时不必给出实参的构造函数,称为调用构造函数时不必给出实参的构造函数,称为 默认构造函数默认构造函数(default cons
22、tructor)。显然,无参的构显然,无参的构 造函数属于默认构造函数。一个类只能有一个默认造函数属于默认构造函数。一个类只能有一个默认 构造函数。构造函数。 (2) 如果在建立对象时选用的是无参构造函数,应注如果在建立对象时选用的是无参构造函数,应注 意正确书写定义对象的语句。意正确书写定义对象的语句。 (3) 尽管在一个类中可以包含多个构造函数,但是对尽管在一个类中可以包含多个构造函数,但是对 于每一个对象来说,建立对象时于每一个对象来说,建立对象时只执行其中一个构只执行其中一个构 造函数造函数,并非每个构造函数都被执行。,并非每个构造函数都被执行。 构造函数中参数的值既可以通过实参传递,
23、也可以构造函数中参数的值既可以通过实参传递,也可以 指定为某些默认值,即如果用户指定为某些默认值,即如果用户不指定实参值不指定实参值,编,编 译系统就译系统就使形参取默认值使形参取默认值。 前面介绍过在函数中可以使用有默认值的参数。在前面介绍过在函数中可以使用有默认值的参数。在 构造函数中也可以采用这样的方法来实现初始化。构造函数中也可以采用这样的方法来实现初始化。 9.1.6 使用默认参数的构造函数使用默认参数的构造函数 在例在例9.3程序的基础上改写如下:程序的基础上改写如下: #include using namespace std; class Box public: Box(int
24、h=10,int w=10,int len=10); /在声明构造函数时指定默认参数在声明构造函数时指定默认参数 int volume( ); private: int height; int width; int length; ; Box Box(int h,int w,int len) /在定义函数时可以不指定默认参数在定义函数时可以不指定默认参数 height=h; width=w; length=len; int Box volume( ) return(height*width*length); int main( ) Box box1; /没有给实参没有给实参 coutThe v
25、olume 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 box3 is box3.volume( )endl; Box box4(15,30,20); /给定给定3个实参个实参 coutThe volume of box4 is box4.volume( )endl; return 0; 程序运行结果为程序运行结果为 The v
26、olume of box1 is 1000 The volume of box2 is 1500 The volume of box3 is 4500 The volume of box4 is 9000 程序中对构造函数的定义程序中对构造函数的定义(第第1216行行)也可以改写成也可以改写成 参数初始化表的形式:参数初始化表的形式: Box Box(int h,int w,int len):height(h),width(w),length(len) 可以看到:可以看到: 在构造函数中使用默认参数是方便而有在构造函数中使用默认参数是方便而有 效的,它提供了建立对象时的多种选择,它的效的,它提
27、供了建立对象时的多种选择,它的作用作用 相当于好几个重载的构造函数。相当于好几个重载的构造函数。它的好处是:它的好处是: 即使即使 在调用构造函数时没有提供实参值,不仅不会出错,在调用构造函数时没有提供实参值,不仅不会出错, 而且还确保按照默认的参数值对对象进行初始化。而且还确保按照默认的参数值对对象进行初始化。 尤其在希望对每一个对象都有同样的初始化状况时尤其在希望对每一个对象都有同样的初始化状况时 用这种方法更为方便。用这种方法更为方便。 说明:说明: (1) 应该在应该在声明构造函数时指定默认值声明构造函数时指定默认值,而不能只在,而不能只在 定义构造函数时指定默认值。定义构造函数时指定
28、默认值。 (2) 程序第程序第5行在声明构造函数时,形参名可以省略。行在声明构造函数时,形参名可以省略。 (3) 如果构造函数的全部参数都指定了默认值,则在如果构造函数的全部参数都指定了默认值,则在 定义对象时可以给一个或几个实参,也可以不给出定义对象时可以给一个或几个实参,也可以不给出 实参。实参。 (4) 在一个类中定义了全部是默认参数的构造函数后,在一个类中定义了全部是默认参数的构造函数后, 不能再定义重载构造函数不能再定义重载构造函数。 析构函数析构函数(destructor)也是一个特殊的成员函数,它也是一个特殊的成员函数,它 的作用与构造函数相反,它的名字是的作用与构造函数相反,它
29、的名字是类名的前面加类名的前面加 一个一个“”符号符号。在在C+中中“”是是位取反运算符位取反运算符, 从这点也可以想到:从这点也可以想到: 析构函数是与构造函数作用相析构函数是与构造函数作用相 反的函数。反的函数。 当对象的生命期结束时,会自动执行析构函数当对象的生命期结束时,会自动执行析构函数。具具 体地说如果出现体地说如果出现以下几种情况以下几种情况,程序就会执行析构,程序就会执行析构 函数:函数: 如果在一个函数中定义了一个对象如果在一个函数中定义了一个对象(它是自它是自 动动局部对象局部对象),当这个,当这个函数被调用结束时函数被调用结束时,对象应该,对象应该 释放,在对象释放前自动
30、执行析构函数。释放,在对象释放前自动执行析构函数。 9.2 析构函数析构函数 static局部对象局部对象在函数调用结束时对象并不释放,在函数调用结束时对象并不释放, 因此也不调用析构函数,只在因此也不调用析构函数,只在main函数结束函数结束或或调用调用 exit函数结束程序时函数结束程序时,才调用,才调用static局部对象的析构函局部对象的析构函 数。数。如果定义了一个如果定义了一个全局对象全局对象,则在,则在程序的流程程序的流程 离开其作用域时离开其作用域时(如如main函数结束或调用函数结束或调用exit函数函数) 时,时, 调用该全局对象的析构函数。调用该全局对象的析构函数。如果用
31、如果用new运算符运算符动动 态地建立了一个对象,当用态地建立了一个对象,当用delete运算符释放该对象运算符释放该对象 时,先调用该对象的析构函数时,先调用该对象的析构函数。 析构函数的作用并不是删除对象,而是在撤销对象析构函数的作用并不是删除对象,而是在撤销对象 占用的内存之前完成一些清理工作,使这部分内存占用的内存之前完成一些清理工作,使这部分内存 可以被程序分配给新对象使用。程序设计者事先设可以被程序分配给新对象使用。程序设计者事先设 计好析构函数,以完成所需的功能,只要对象的生计好析构函数,以完成所需的功能,只要对象的生 命期结束,程序就自动执行析构函数来完成这些工命期结束,程序就
32、自动执行析构函数来完成这些工 作。作。 析构函数不返回任何值,没有函数类型,也没有函析构函数不返回任何值,没有函数类型,也没有函 数参数。它不能被重载。一个类可以有多个构造函数参数。它不能被重载。一个类可以有多个构造函 数,但只能有一个析构函数。数,但只能有一个析构函数。 实际上,析构函数的作用并不仅限于释放资源方面,实际上,析构函数的作用并不仅限于释放资源方面, 它还可以被用来执行它还可以被用来执行“用户希望在最后一次使用对用户希望在最后一次使用对 象之后所执行的任何操作象之后所执行的任何操作”,例如输出有关的信息。例如输出有关的信息。 这里说的用户是指类的设计者,因为,析构函数是这里说的用
33、户是指类的设计者,因为,析构函数是 在声明类的时候定义的。也就是说,析构函数可以在声明类的时候定义的。也就是说,析构函数可以 完成类的设计者所指定的任何操作。完成类的设计者所指定的任何操作。 一般情况下,一般情况下,类的设计者应当在声明类的同时定义类的设计者应当在声明类的同时定义 析构函数,以指定如何完成析构函数,以指定如何完成“清理清理”的工作。的工作。如果如果 用户没有定义析构函数,用户没有定义析构函数,C+编译系统会自动生成一编译系统会自动生成一 个析构函数,但它只是徒有析构函数的名称和形式,个析构函数,但它只是徒有析构函数的名称和形式, 实际上什么操作都不进行。想让析构函数完成任何实际
34、上什么操作都不进行。想让析构函数完成任何 工作,都必须在定义的析构函数中指定。工作,都必须在定义的析构函数中指定。 例例9.5 包含构造函数和析构函数的包含构造函数和析构函数的C+程序。程序。 #include #include using namespace std; class Student /声明声明Student类类 public: student(int n,string nam,char s ) /定义构造函数定义构造函数 num=n; name=nam; sex=s; coutConstructor called.endl; /输出有关信息输出有关信息 Student( ) /
35、定义析构函数定义析构函数 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); /建立对象建立对象stud1 stud1.display( ); /输出学生输出学生1的数据的数据 Student stud2(1001
36、1,Zhang_fun,m); /定义对象定义对象stud2 stud2.display( ); /输出学生输出学生2的数据的数据 return 0; 程序运行结果如下:程序运行结果如下: Constructor called. (执行执行stud1的构造函数的构造函数) num: 10010 (执行执行stud1的的display函数函数) name:Wang_li sex: f Constructor called. (执行执行stud2的构造函数的构造函数) num: 10011 (执行执行stud2的的display函数函数) name:Zhang_fun sex:m Destruct
37、or called. (执行执行stud2的析构函数的析构函数) Destructor called. (执行执行stud1的析构函数的析构函数) 在使用构造函数和析构函数时,需要在使用构造函数和析构函数时,需要特别注意对它特别注意对它 们的调用时间和调用顺序们的调用时间和调用顺序。 在一般情况下,在一般情况下,调用析构函数的次序正好与调用构调用析构函数的次序正好与调用构 造函数的次序相反:造函数的次序相反: 最先被调用的构造函数,其对最先被调用的构造函数,其对 应的应的(同一对象中的同一对象中的)析构函数最后被调用,而最后被析构函数最后被调用,而最后被 调用的构造函数,其对应的析构函数最先被
38、调用。调用的构造函数,其对应的析构函数最先被调用。 如图如图9.1示意。示意。 9.3 调用构造函数和析构函数的顺序调用构造函数和析构函数的顺序 图图9.1 但是,并不是在任何情况下都是按这一原则处理的。但是,并不是在任何情况下都是按这一原则处理的。 在第在第4章第章第4.11和和4.12节中曾介绍过作用域和存储类别节中曾介绍过作用域和存储类别 的概念,这些概念对于对象也是适用的。对象可以的概念,这些概念对于对象也是适用的。对象可以 在不同的作用域中定义,可以有不同的存储类别。在不同的作用域中定义,可以有不同的存储类别。 这些会影响调用构造函数和析构函数的时机。这些会影响调用构造函数和析构函数
39、的时机。 下面归纳一下什么时候调用构造函数和析构函数:下面归纳一下什么时候调用构造函数和析构函数: (1) 在在全局范围中定义的对象全局范围中定义的对象(即在所有函数之外定义即在所有函数之外定义 的对象的对象),它的构造函数在文件中的所有函数,它的构造函数在文件中的所有函数(包括包括 main函数函数)执行之前调用。但如果一个程序中有多个执行之前调用。但如果一个程序中有多个 文件,而不同的文件中都定义了全局对象,则这些文件,而不同的文件中都定义了全局对象,则这些 对象的构造函数的执行顺序是不确定的。对象的构造函数的执行顺序是不确定的。当当main函函 数执行完毕或调用数执行完毕或调用exit函
40、数时函数时(此时程序终止此时程序终止),调用,调用 析构函数。析构函数。 (2) 如果定义的是如果定义的是局部自动对象局部自动对象(例如在函数中定义对例如在函数中定义对 象象),则在建立对象时调用其构造函数。如果函数被,则在建立对象时调用其构造函数。如果函数被 多次调用,则在每次建立对象时都要调用构造函数。多次调用,则在每次建立对象时都要调用构造函数。 在函数调用结束、对象释放时先调用析构函数。在函数调用结束、对象释放时先调用析构函数。 (3) 如果在函数中定义如果在函数中定义静态静态(static)局部对象局部对象,则只在,则只在 程序第一次调用此函数建立对象时调用构造函数一程序第一次调用此
41、函数建立对象时调用构造函数一 次,在调用结束时对象并不释放,因此也不调用析次,在调用结束时对象并不释放,因此也不调用析 构函数,只构函数,只在在main函数结束或调用函数结束或调用exit函数结束程序函数结束程序 时,才调用析构函数时,才调用析构函数。 构造函数和析构函数在面向对象的程序设计中是相构造函数和析构函数在面向对象的程序设计中是相 当重要的。以上介绍了最基本的、使用最多的普通当重要的。以上介绍了最基本的、使用最多的普通 构造函数,在本章第构造函数,在本章第9.8节中将会介绍节中将会介绍复制构造函数复制构造函数, 在第在第10章第章第10.7节中还要介绍节中还要介绍转换构造函数转换构造
42、函数。 数组不仅可以由简单变量组成数组不仅可以由简单变量组成(例如整型数组的每一例如整型数组的每一 个元素都是整型变量个元素都是整型变量),也可以由对象组成也可以由对象组成(对象数组对象数组 的每一个元素都是同类的对象的每一个元素都是同类的对象)。 在日常生活中,有许多实体的属性是共同的,只是在日常生活中,有许多实体的属性是共同的,只是 属性的具体内容不同。例如一个班有属性的具体内容不同。例如一个班有50个学生,每个学生,每 个学生的属性包括姓名、性别、年龄、成绩等。如个学生的属性包括姓名、性别、年龄、成绩等。如 果为每一个学生建立一个对象,需要分别取果为每一个学生建立一个对象,需要分别取50
43、个对个对 象名。用程序处理很不方便。这时可以定义一个象名。用程序处理很不方便。这时可以定义一个 “学生类学生类”对象数组,每一个数组元素是一个对象数组,每一个数组元素是一个“学学 生类生类”对象。例如对象。例如 Student stud50; /假设已声明了假设已声明了Student类,定义类,定义stud数组,有数组,有50个元素个元素 9.4 对象数组对象数组 在建立数组时,同样要调用构造函数。如果有在建立数组时,同样要调用构造函数。如果有50个个 元素,需要调用元素,需要调用50次构造函数。次构造函数。在需要时可以在定在需要时可以在定 义数组时提供实参以实现初始化。义数组时提供实参以实现
44、初始化。如果构造函数如果构造函数只只 有一个参数有一个参数,在定义数组时可以直接在等号后面的,在定义数组时可以直接在等号后面的 花括号内提供实参。花括号内提供实参。如如 Student stud3=60,70,78; /合法,合法,3个实参分别传递给个实参分别传递给3个数组元素的构造函个数组元素的构造函 数数 如果构造函数如果构造函数有多个参数有多个参数,则不能用在定义数组时,则不能用在定义数组时 直接提供所有实参的方法,直接提供所有实参的方法,因为一个数组有多个元因为一个数组有多个元 素,对每个元素要提供多个实参,如果再考虑到构素,对每个元素要提供多个实参,如果再考虑到构 造函数有造函数有默
45、认参数默认参数的情况,很容易造成实参与形参的情况,很容易造成实参与形参 的对应关系不清晰,出现歧义性。例如,类的对应关系不清晰,出现歧义性。例如,类Student 的构造函数有多个参数,且为的构造函数有多个参数,且为默认参数默认参数: Student Student(int=1001,int=18,int=60); /定义构造函数,有多个参数,且为定义构造函数,有多个参数,且为 默认参数默认参数 如果定义对象数组的语句为如果定义对象数组的语句为 Student stud3=1005,60,70; 编译系统编译系统只为每个对象元素的构造函数传递一个实只为每个对象元素的构造函数传递一个实 参,参,
46、所以在定义数组时提供的所以在定义数组时提供的实参个数不能超过数实参个数不能超过数 组元素个数组元素个数,如如 Student stud3=60,70,78,45; /不合法,实参个数超过对象数组元素个数不合法,实参个数超过对象数组元素个数 那么,那么,如果构造函数有多个参数,在定义对象数组如果构造函数有多个参数,在定义对象数组 时应当怎样实现初始化呢?时应当怎样实现初始化呢?回答是:回答是: 在花括号中分在花括号中分 别写出构造函数并指定实参。别写出构造函数并指定实参。如果构造函数有如果构造函数有3个参个参 数,分别代表学号、年龄、成绩。则可以这样定义数,分别代表学号、年龄、成绩。则可以这样定
47、义 对象数组:对象数组: Student Stud3= /定义对象数组定义对象数组 Student(1001,18,87), /调用第调用第1个元素的构造函数,为它提供个元素的构造函数,为它提供3个实参个实参 Student(1002,19,76), /调用第调用第2个元素的构造函数,为它提供个元素的构造函数,为它提供3个实参个实参 Student(1003,18,72) /调用第调用第3个元素的构造函数,为它提供个元素的构造函数,为它提供3个实参个实参 ; 在建立对象数组时,分别调用构造函数,对每个元在建立对象数组时,分别调用构造函数,对每个元 素初始化。每一个元素的实参分别用括号包起来,素
48、初始化。每一个元素的实参分别用括号包起来, 对应构造函数的一组形参,不会混淆对应构造函数的一组形参,不会混淆。 例例9.6 对象数组的使用方法。对象数组的使用方法。 #include using namespace std; class Box public: Box(int h=10,int w=12,int len=15): height(h),width(w),length(len) /声明有默认参数的构造函数,用参数初始化表对数据成员初始化声明有默认参数的构造函数,用参数初始化表对数据成员初始化 int volume( ); private: int height; int width
49、; int length; ; int Box volume( ) return(height*width*length); int main( ) Box a3= /定义对象数组定义对象数组 Box(10,12,15), /调用构造函数调用构造函数Box,提供第提供第1个元素的实参个元素的实参 Box(15,18,20), /调用构造函数调用构造函数Box,提供第提供第2个元素的实参个元素的实参 Box(16,20,26) /调用构造函数调用构造函数Box,提供第提供第3个元素的实参个元素的实参 ; coutvolume of a0 is a0.volume( )endl; /调用调用a0的
50、的volume函数函数 coutvolume of a1 is a1.volume( )endl; /调用调用a1 的的volume函数函数 coutvolume of a2 is a2.volume( )endl; /调用调用a2 的的volume函数函数 运行结果如下:运行结果如下: volume of a0 is 1800 volume of a1 is 5400 volume of a2 is 8320 在建立对象时,编译系统会为每一个对象分配一定在建立对象时,编译系统会为每一个对象分配一定 的存储空间,以存放其成员。的存储空间,以存放其成员。对象空间的起始地址对象空间的起始地址 就是
51、对象的指针。就是对象的指针。可以定义一个指针变量,用来存可以定义一个指针变量,用来存 放对象的指针。如果有一个类:放对象的指针。如果有一个类: class Time public: int hour; int minute; int sec; void get_time( ); ; 9.5 对象指针对象指针 9.5.1 指向对象的指针指向对象的指针 void Time get_time( ) couthour:minute:sechour pt所指向的对象中的所指向的对象中的hour成员,即成员,即t1.hour (*pt).get_time ( ) 调用调用pt所指向的对象中的所指向的对象中
52、的get_time函数,即函数,即t1.get_time pt-get_time ( ) 调用调用pt所指向的对象中的所指向的对象中的get_time函数,即函数,即t1.get_time 对象有地址,对象有地址,存放对象初始地址的指针变量存放对象初始地址的指针变量就是就是指指 向对象的指针变量向对象的指针变量。对象中的成员也有地址,。对象中的成员也有地址,存放存放 对象成员地址的指针变量对象成员地址的指针变量就是就是指向对象成员的指针指向对象成员的指针 变量。变量。 9.5.2 指向对象成员的指针指向对象成员的指针 1. 指向对象数据成员的指针指向对象数据成员的指针 定义指向对象数据成员的指
53、针变量的方法和定义指定义指向对象数据成员的指针变量的方法和定义指 向普通变量的指针变量方法相同。例如向普通变量的指针变量方法相同。例如 int *p1; /定义指向整型数据的指针变量定义指向整型数据的指针变量 定义指向对象数据成员的指针变量的一般形式为定义指向对象数据成员的指针变量的一般形式为 数据类型名数据类型名 *指针变量名;指针变量名; 如果如果Time类的数据成员类的数据成员hour为公用的整型数据,则为公用的整型数据,则 可以在类外通过指向对象数据成员的指针变量访问可以在类外通过指向对象数据成员的指针变量访问 对象数据成员对象数据成员hour。 p1= /将对象将对象t1的数据成员的
54、数据成员hour的地址赋给的地址赋给p1,p1指向指向t1.hour cout*p1endl; /输出输出t1.hour的值的值 2. 指向对象成员函数的指针指向对象成员函数的指针 定义指向对象成员函数的指针变量的方法和定义指定义指向对象成员函数的指针变量的方法和定义指 向普通函数的指针变量方法有所不同。向普通函数的指针变量方法有所不同。 成员函数与普通函数有一个最根本的区别:成员函数与普通函数有一个最根本的区别: 它是类它是类 中的一个成员。中的一个成员。编译系统要求在上面的赋值语句中,编译系统要求在上面的赋值语句中, 指针变量的类型必须与赋值号右侧函数的类型相匹指针变量的类型必须与赋值号右
55、侧函数的类型相匹 配,要求在以下配,要求在以下3方面都要匹配:方面都要匹配: 函数参数的类型函数参数的类型 和参数个数;函数返回值的类型;所属的类。和参数个数;函数返回值的类型;所属的类。 定义指向成员函数的指针变量采用下面的形式:定义指向成员函数的指针变量采用下面的形式: void (Time *p2)( ); /定义定义p2为指向为指向Time类中公用成员函数的指针变量类中公用成员函数的指针变量 定义指向公用成员函数的指针变量的一般形式为定义指向公用成员函数的指针变量的一般形式为 数据类型名数据类型名 (类名类名 *指针变量名指针变量名)(参数表列参数表列); 可以让它指向一个公用成员函数
56、,只需把公用成员可以让它指向一个公用成员函数,只需把公用成员 函数的入口地址赋给一个指向公用成员函数的指针函数的入口地址赋给一个指向公用成员函数的指针 变量即可。如变量即可。如 p2= 使指针变量指向一个公用成员函数的一般形式为使指针变量指向一个公用成员函数的一般形式为 指针变量名指针变量名= 例例9.7 有关对象指针的使用方法。有关对象指针的使用方法。 #include using namespace std; class Time public: Time(int,int,int); int hour; int minute; int sec; void get_time( ); /声明公
57、有成员函数声明公有成员函数 ; Time Time(int h,int m,int s) hour=h; minute=m; sec=s; void Time get_time( ) /定义公有成员函数定义公有成员函数 couthour:minute: secendl; int main( ) Time t1(10,13,56); /定义定义Time类对象类对象t1 int *p1= /定义指向整型数据的指针变量定义指向整型数据的指针变量p1,并使并使p1指向指向t1.hour cout*p1get_time( ); /调用调用p2所指向对象所指向对象(即即t1)的的get_time函数函数
58、void (Time *p3)( ); /定义指向定义指向Time类公用成员函数的指针变量类公用成员函数的指针变量p3 p3= /使使p3指向指向Time类公用成员函数类公用成员函数get_time (t1.*p3)( ); /调用对象调用对象t1中中p3所指的成员函数所指的成员函数(即即t1.get_time( ) 程序运行结果为程序运行结果为 10 (main函数第函数第4行的输出行的输出) 10:13:56 (main函数第函数第5行的输出行的输出) 10:13:56 (main函数第函数第7行的输出行的输出) 10:13:56 (main函数第函数第10行的输出行的输出) 可以看到为了
59、输出可以看到为了输出t1中中hour,minute和和sec的值,可以的值,可以 采用采用3种不同的方法。种不同的方法。 说明:说明: (1) 从从main函数第函数第9行可以看出:行可以看出: 成员函数的入口地成员函数的入口地 址的正确写法是:址的正确写法是: /定义指针变量时定义指针变量时 指定其指向指定其指向 前面曾经提到过:前面曾经提到过: 每个对象中的数据成员都分别占每个对象中的数据成员都分别占 有存储空间,如果对同一个类定义了有存储空间,如果对同一个类定义了n个对象,则有个对象,则有 n组同样大小的空间以存放组同样大小的空间以存放n个对象中的数据成员。个对象中的数据成员。 但是,不
60、同对象都调用同一个函数代码段。但是,不同对象都调用同一个函数代码段。 那么,当不同对象的成员函数引用数据成员时,怎那么,当不同对象的成员函数引用数据成员时,怎 么能保证引用的是所指定的对象的数据成员呢?假么能保证引用的是所指定的对象的数据成员呢?假 如,对于例如,对于例9.6程序中定义的程序中定义的Box类,定义了类,定义了3个同类个同类 对象对象a,b,c。如果有如果有a.volume( ),应该是引用对象应该是引用对象a中中 的的height,width和和length,计算出长方体计算出长方体a的体积。的体积。 如果有如果有b.volume( ),应该是引用对象应该是引用对象b中的中的h
温馨提示
- 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
- 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
- 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
- 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
- 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
- 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
- 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。
最新文档
- 2024年食品质检员的考试结构与答案
- 2024年汽车美容行业的新兴技术试题及答案
- 2024年电瓶检测与更换流程试题及答案
- 汽车美容师市场营销与客户需求识别试题及答案
- 心理健康教育活动课
- 果汁店创新创业计划书
- 2024年公务员省考与汽车行业方位试题及答案
- 汽车售后服务发展的重要性分析试题及答案
- 2024年汽车维修工电工基础知识试题及答案
- 2024年汽车美容师客户反馈处理试题及答案
- 国家网络安全知识竞赛题库附参考答案(综合卷)
- 网课智慧树知道《人工智能引论(浙江大学)》章节测试答案
- 2024年南通市高考《数学》第四次模拟试卷(含答案)
- WD-PSO-LSTM模型在光伏出力预测中的应用
- 期中测试卷(试题)-2023-2024学年六年级下册数学苏教版
- 广东省深圳市2023-2024学年六年级下册(全册)期中模拟测试数学试卷(北师大版)
- 安全教育普法
- 分层过程审核培训-课后测试附有答案
- 江苏省南京市鼓楼区2022-2023学年五年级下学期期中语文试卷
- 幼儿园优质公开课:小班综合《小鸡过生日》课件
- 《新媒体推广》项目二图文推广-课前自学
评论
0/150
提交评论