Download presentation
Presentation is loading. Please wait.
1
C++语言程序设计教程 第7章 类与对象 第7章 类与对象
2
第7章 类与对象 学习目标 1. 掌握类的定义,会根据需求设计类; 2. 会根据类创建各种对象; 3. 掌握对象的各种成员的使用方法;
C++语言程序设计教程 第7章 类与对象 第7章 类与对象 学习目标 1. 掌握类的定义,会根据需求设计类; 2. 会根据类创建各种对象; 3. 掌握对象的各种成员的使用方法; 4. 会设计构造函数与拷贝构造函数来初始化对象;理解其调用过程与顺序; 5. 理解浅拷贝与深拷贝的概念; 6. 掌握动态对象以及动态对象数组的建立与释放。 7. 理解类的静态成员的概念; 8. 理解友元函数与友元类的概念; 9. 掌握常对象与常成员的使用; 10.了解对象在内存中的分布情况。
3
7.1 类与对象 在面向对象的程序设计中,程序模块是由类构成的。类是对逻辑上相关的函数与数据的封装,它是对问题的抽象描述。
C++语言程序设计教程 第7章 类与对象 7.1 类与对象 在面向过程的结构化程序设计中,程序模块是由函数构成,函数将对数据处理的语句放在函数体内,完成特定的功能,数据则通过函数参数传递进入函数体。 在面向对象的程序设计中,程序模块是由类构成的。类是对逻辑上相关的函数与数据的封装,它是对问题的抽象描述。 面向对象的程序设计方法就是运用面向对象的观点来对现实世界中的各种问题进行抽象,然后用计算机程序来描述并解决该问题,这种描述和处理是通过类与对象实现的。 类与对象是C++程序设计中最重要的概念
4
C++语言程序设计教程 第7章 类与对象 7.1 类与对象 【例7-1】 模拟时钟 分析:不管什么样的时钟, 也不管各种时钟是如何运行的, 它都具有时、分、秒3个属性。除了运行、显示时间的基本功能外,还有设置(调整)时间、设置闹钟等功能。将时钟的这些属性与功能抽象出来,分别给出面向过程的程序与面向对象的程序来实现对时钟的模拟。 思考:二者有何不同?
5
7.1.2 类的定义 说明: 简单讲,类是一个包含函数的结构体。因此,类的定义与结构类型的定义相似,其格式如下: class 类名 {
第7章 类与对象 类的定义 简单讲,类是一个包含函数的结构体。因此,类的定义与结构类型的定义相似,其格式如下: class 类名 { public: 公有数据成员或公有函数成员的定义; protected: 保护数据成员或保护函数成员的定义; private: 私有数据成员或私有函数成员的定义; }; 说明: 关键字class表明定义的是一个类; 类名是类的名称,应是一个合法的标识符; public、protected、private为存取控制属性(访问权限),用来控制对类的 成员的存取。如果前面没有标明访问权限,默认访问权限为private。 类的成员有数据成员与函数成员两类,类的数据成员和函数成员统称为类的 成员,类的数据成员一般用来描述该类对象的属性,称为属性;函数成员是描 述类行为,称作方法。函数成员由函数构成,这些作为类成员的函数因此也 叫成员函数。
6
程序解释: 运行结果: 0:0:0 -85893460:-85893460:-5893460 8:30:30 8:30:30
C++语言程序设计教程 第7章 类与对象 ***************************** * p7_1_b.cpp * * 面向对象的时钟程序 * ******************************/ #include <iostream> using namespace std; class Clock { private: int H,M,S; public: void SetTime(int h,int m,int s) { H=(h>=0&&h<24)?h:0; M=(m>=0&&m<60)?m:0; S=(s>=0&&s<60)?s:0; } void ShowTime() cout<<H<<":"<<M<<":"<<S<<endl; }; void main(void) { Clock MyClock; MyClock.ShowTime(); MyClock.SetTime(8,30,30); /***************************** * p7_1_a.cpp * * 面向过程的时钟程序 * struct Clock { Clock MyClock; void SetTime(int H,int M,int S) MyClock.H=(H>=0&&H<24)?H:0; MyClock.M=(M>=0&&M<60)?M:0; MyClock.S=(S>=0&&S<60)?S:0; cout<<MyClock.H<<":"; cout<<MyClock.M<<":"; cout<<MyClock.S<<endl; { ShowTime(); SetTime(8,30,30); ShowTime(); 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 时钟程序B 时钟程序A 程序解释: 通过对上述两种方案的程序进行简单的观察,可以发现它们存在下面几点不同: 在程序A中,时钟数据用一个结构型的变量存储,对时钟数据的存取通过函数实现。由于存 储时钟数据的是一个全局变量,在任何地方都可见,可以不通过函数单独存取时钟数据。 程序B中,只能通过类提供的函数操作时钟。 程序A中,数据与对数据操作相互独立,数据作为参数传递给函数。程序B中,数据与对数 据的操作构成一个整体。 程序A与程序B运行的初始结果不同。这是因为在程序A中,变量是全局的;在程序B中, 对 象(变量)MyClock是函数main()中的局部对象。全局变量与局部变量在没有初始化时,取初 值方式不同,这样造成了运行结果不同。将第23行移出main()外,使之变成全局对象后, 两程序结果完全相同。 在程序B中发现,一个以class开头的类似结构体的结构,将时钟的数据与对数据进行处理 的函数包括在一起,这就是用C++实现的类。 运行结果: 0:0: : : 8:30: :30:30
7
C++语言程序设计教程 第7章 类与对象 类的定义 例如:例7-1中定义了一个时钟类Clock。
8
7.1.2 类的定义 类定义中的数据成员描述了类对象所包含的数据类型,数据成员的类型可以是C++基本数据类型,也可以是构造数据类型。
第7章 类与对象 类的定义 1. 数据成员 类定义中的数据成员描述了类对象所包含的数据类型,数据成员的类型可以是C++基本数据类型,也可以是构造数据类型。 struct Record { char name[20]; int score; }; class Team { private: int num; //基本数据类型 Record *p; //构造数据类型
9
7.1.2 类的定义 注意: class Team; //已定义的类 class Grade { Team a; // 使用了已定义的类类型
第7章 类与对象 类的定义 class Team; //已定义的类 class Grade { Team a; // 使用了已定义的类类型 Grade *p; // 使用正在定义的类类型定义指针成员 Grade &r; // 使用正在定义的类类型定义引用成员 Grade b; // 错误! 使用了未定义完的类Record 定义变量 }; 注意: 因为类只是一种类型,类中的数据成员不占内存空间,因此在定义数据成员 时不能给数据成员赋初值。 类的数据成员除了可以使用前面讲述的C++类型外,还可以使用已定义完整的 类类型。 在正在定义的类中,由于该类型没有定义完整,所以不能定义该类类型的变 量,只能定义该类类型的指针成员以及该类类型的引用成员。
10
C++语言程序设计教程 第7章 类与对象 类的定义 2. 成员函数 作为类成员的成员函数描述了对类中的数据成员实施的操作。成员函数的定义、声明格式与非成员函数(全局函数)的格式相同。成员函数可以放在类中定义,也可以放在类外。放在类中定义的成员函数为内联(inline)函数。 Clock类中的成员函数就是放在类内定义的。 C++可以在类内声明成员函数的原型,在类外定义函数体。这样做的好处是相当于在类内列了一个函数功能表,使我们对类的成员函数的功能一目了然,避免了在各个函数实现的大堆代码中查找函数的定义。在类中声明函数原型的方法与一般函数原型的声明一样,在类外定义函数体的格式如下: ::是类的作用域分辨符,用在此处,放在类名后成员函数前,表明后面的成员函数属于前面的那个类。 返回值类型 类名 :: 成员函数名(形参表) { 函数体; }
11
7.1.2 类的定义 C++语言程序设计教程 Clock类中的成员函数可以在类中声明: class Clock { private: 第7章
类与对象 类的定义 Clock类中的成员函数可以在类中声明: class Clock { private: int H,M,S; public: void SetTime(int h,int m,int s); //声明成员函数 void ShowTime(); //声明成员函数 }; 在类外实现成员函数如下: void Clock::SetTime(int h,int m,int s) { H=h,M=m,S=s; }; void Clock:: ShowTime() cout<<H<<":"<<M<<":"<<S<<endl; }
12
C++语言程序设计教程 第7章 类与对象 类的定义 如果要将类外定义的成员函数编译成内联函数,可以在类外定义函数时,函数的返回类型前加上inline; 下面将ShowTime()定义成内联函数,与在类内定义成员函数的效果相同。 inline void Clock::ShowTime() { cout<<H<<":"<<M<<":"<<S<<endl; } 3. 类作用域 类是一组数据成员和函数成员的集合,类作用域作用于类中定义的特定的成员,包括数据成员与成员函数,类内的每一个成员都具有类作用域。实际上,类的封装作用也就是限制类的成员其访问范围局限于类的作用域之内。
13
7.1.2 类的定义 注意: class Clock { private: int H,M,S; public:
第7章 类与对象 类的定义 class Clock { private: int H,M,S; public: Clock AddTime(Clock C2){ // 形参为Clock类型的变量 Clock T; // 函数体中定义了Clock类型的变量 ... return T; // 返回类型为Clock类型 } }; 注意: 在成员函数中不仅可以自由使用类的成员,还可以使用该类定义变量(对象),通过变量使用成员。其原因是函数在调用时才在栈内存中建立函数体中的变量(包括实参),这时类已经定义完毕,当然可以使用已定义完整的类类型的变量。
14
C++语言程序设计教程 第7章 类与对象 对象的建立与使用 类相当于一种包含函数的自定义数据类型,它不占内存,是一个抽象的“虚”体,使用已定义的类建立对象就像用数据类型定义变量一样。对象建立后,对象占据内存,变成了一个“实”体。类与对象的关系就像数据类型与变量的关系一样。其实,一个变量就是一个简单的不含成员函数的数据对象。 类名 对象名; 建立对象的格式如下: 其中,对象名可以是简单的标识符,也可以是数组。 在例7-1中,使用: Clock MyClock; 建立了一个Clock型的对象MyClock。 类名 对象名;
15
7.1.3 对象的建立与使用 注意: 在建立对象后,就可以通过对象存取对象中的数据成员,调用成员函数。存取语法如下: 对象名.属性
C++语言程序设计教程 第7章 类与对象 对象的建立与使用 在建立对象后,就可以通过对象存取对象中的数据成员,调用成员函数。存取语法如下: 例如,通过对象MyClock使用成员函数SetTime()的方式如下: MyClock.SetTime(8,30,30); 至于对数据成员H、M、S的存取,因其存取权限为pravate而被保护, 所以不能进行直接存取。 对象名.属性 对象名.成员函数名(实参1, 实参2,…,) 注意: 为节省内存,编译器在创建对象时,只为各对象分配用于保存各对象数据成员初始化的值,并不为各对象的成员函数分配单独的内存空间,而是共享类的成员函数定义,即类中成员函数的定义为该类的所有对象所共享,这是C++编译器创建对象的一种方法,在实际应用中,我们仍要将对象理解为由数据成员和函数成员两部分组成。
16
7.1.4 成员的存取控制 public 公开(公有)级 该类成员以及所有对象 protected 保护级 该类及其子类成员 private
第7章 类与对象 7.1.4 成员的存取控制 通过设置成员的存取控制属性, 使对类成员的存取得到控制,从而达到了信息隐藏的目的。C++的存取控制属性有:公有类型(public)、私有类型(private)和保护类型(protected)。三者的意义如下: 表7-1 存取控制属性表 存取属性 意 义 可存取对象 public 公开(公有)级 该类成员以及所有对象 protected 保护级 该类及其子类成员 private 私有级 该类的成员 类中定义为public等级的成员,可以被该类的任何对象存取,适用于完全公开的数据。而private等级的成员只可被类内的成员存取,适用于不公开的数据。至于protected等级,属于半公开性质的数据,定义为protected等级的成员,可以被该类及其子类存取。关于子类的概念,在以后的章节中讲述。
17
C++语言程序设计教程 第7章 类与对象 7.1.4 成员的存取控制 在Clock类中,H、M、S的存取控制属性为privated。这样,这些数据不能在类外存取而被保护,下列存取方法是错误的: MyClock.M=30; 而成员函数SetTime()、ShowTime()存取控制属性为public, 因此在类外可以通过对象存取。 由于private成员被隐藏起来,不能直接在类外被存取,为了取得这些被隐藏的数据,通常在类内定义一个public的成员函数,通过该成员函数存取private成员,而public的成员函数又能在类外被调用。这样通过调用public型的成员函数,间接存取到private成员。这样的函数起到了为private成员提供供外界访问的接口作用。 类Clock中成员函数SetTime()、ShowTime()就是存取private数据成员H、M、S的接口。 通过接口访问类的数据成员,一方面有效保护数据成员,另一方面又保证了数据的合理性。
18
C++语言程序设计教程 第7章 类与对象 7.2 构造函数与析构函数 在定义一个对象的同时,希望能给它的数据成员赋初值――对象的初始化。在特定对象使用结束时,还经常需要进行一些清理工作。C++程序中的初始化和清理工作分别由两个特殊的成员函数来完成,它们就是构造函数和析构函数。
19
C++语言程序设计教程 第7章 类与对象 7.2.1 构造函数 构造函数(constructor) 构造函数是与类名相同的,在建立对象时自动调用的函数。如果在定义类时,没有为类定义构造函数,编译系统就生成一个默认形式的隐含的的构造函数,这个构造函数的函数体是空的,因此默认构造函数不具备任何功能。 如果用户至少为类定义了一个构造函数,C++就不会生成任何默认的构造函数,而是根据对象的参数类型和个数从用户定义的构造函数中选择最合适的构造函数完成对该对象的初始化。 作为类的成员函数,构造函数可以直接访问类的所有数据成员,可以是内联函数,可以不带任何参数,可以带有参数表以及默认形参值,还可以重载,用户可以根据不同问题的具体需要,有针对性地设计合适的构造函数将对象初始化为特定的状态。
20
7.2.1 构造函数 例如,将例7-1 程序p7-1_b.cpp中的Clock类中添加带有默认形参值的构造函数:
第7章 类与对象 7.2.1 构造函数 例如,将例7-1 程序p7-1_b.cpp中的Clock类中添加带有默认形参值的构造函数: Clock(int h=0,int m=0,int s=0) { H=(h>=0&&h<24)?h:0; M=(m>=0&&m<60)?m:0; S=(s>=0&&s<60)?s:0; } 执行:Clock MyClock; MyClock.ShowTime(); 显示结果为: 0:0:0 这是因为建立对象时调用了Clock(),各个形参被设成了默认值。 当执行: Clock MyClock(9,30,45); MyClock.ShowTime(); 9:30:45 这是因为建立对象时调用了Clock(9,30,45)
21
7.2.1 构造函数 构造函数的函数名必须与定义它的类同名。 构造函数没有返回值。如果在构造函数前加void是错误的。
C++语言程序设计教程 第7章 类与对象 7.2.1 构造函数 构造函数是类的一个成员函数,除了具有一般成员函数的特征之外,还归纳出如下特殊的性质: 构造函数的函数名必须与定义它的类同名。 构造函数没有返回值。如果在构造函数前加void是错误的。 构造函数被声明定义为公有函数。 构造函数在建立对象时由系统自动调用。 注意:由于构造函数可以重载,可以定义多个构造函数,在建立对象时根据参数来调用相应的构造函数。如果相应的构造函数没有定义,则出错。例如,若定义例7-1中的构造函数,而不是带默认形参值的构造函数: Clock(int h, int m,int s) { H=(h>=0&&h<24)?h:0; M=(m>=0&&m<60)?m:0; S=(s>=0&&s<60)?s:0; } 定义对象Clock MyClock;时,调用Clock(), 而Clock类没有Clock()函数,因而出错。
22
C++语言程序设计教程 第7章 类与对象 7.2.2 析构函数 自然界万物都是有生有灭,程序中的对象也是一样。对象在定义时诞生,不同生存期的对象在不同的时期消失。在对象要消失时,通常有一些善后工作需要做, 例如:构造对象时,通过构造函数动态申请了一些内存单元,在对象消失之前就要释放这些内存单元。C++用什么来保证这些善后清除工作的执行呢?答案是:析构函数。 析构函数(destructor)也译作拆构函数, 是在对象消失之前的瞬间自动调用的函数, 其形式是: 构造函数名(类名)前加上一个逻辑非运算符~,以示与构造函数相反。 析构函数与构造函数的作用几乎正好相反,相当于“逆构造函数”。析构函数也是类的一个特殊的公有函数成员, ~构造函数名();
23
7.2.2 析构函数 析构函数具有以下特点: 析构函数没有任何参数,不能被重载,但可以是虚函数,一个类只有一个析构函数。
C++语言程序设计教程 第7章 类与对象 7.2.2 析构函数 析构函数具有以下特点: 析构函数没有任何参数,不能被重载,但可以是虚函数,一个类只有一个析构函数。 析构函数没有返回值。 析构函数名与类名相同,但在类名前加上一个逻辑非运算符“~”,以示与构造函数对比区别。 析构函数一般由用户自己定义,在对象消失时由系统自动调用,如果用户没有定义析构函数,系统将自动生成一个不做任何事的默认析构函数。 注意: 在对象消失时的清理工作并不是由析构函数完成,而是靠用户在析构函数中添加清理语句完成。
24
{ i=4 i=5 i=3 i=2 } } C++语言程序设计教程 第7章 类与对象 7.2.2 析构函数 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 /***************************** * p7_2.cpp * * 构造函数与析构函数 * *****************************/ #include <iostream> using namespace std; class Clock { private: int H,M,S; public: Clock(int h=0,int m=0,int s=0) { H=h,M=m,S=s; cout<<"constructor:"<<H<<":"<<M<<":"<<S<<endl; }
25
C++语言程序设计教程 第7章 类与对象 7.2.2 析构函数 16 17 18 19 20 21 22 23 24 25 26 27 ~Clock() { cout<<"destructor:"<<H<<":"<<M<<":"<<S<<endl; } }; Clock C1(8,0,0); Clock C2(9,0,0); void main(void) Clock C3(10,0,0); Clock C4(11,0,0); 运行结果: constructor:8:0:0 constructor:9:0:0 constructor:10:0:0 constructor:11:0:0 destructor:11:0:0 destructor:10:0:0
26
/********************************** * p7_3.cpp * * 基本的字符串类 *
第7章 类与对象 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 /********************************** * p7_3.cpp * * 基本的字符串类 * **********************************/ #include <iostream> using namespace std; class String { private: char *Str; int len; public: void ShowStr() { cout<<"string:"<<Str<<",length:"<<len<<endl; } String() len=0; Str=NULL;
27
运行结果: string:ABCDE, length:5 string:123456, length:6 C++语言程序设计教程 第7章
类与对象 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 String(const char *p) { len=strlen(p); Str=new char[len+1]; strcpy(Str,p); } ~String() if (Str!=NULL) delete [] Str; Str=NULL; }; void main(void) { char s[]="ABCDE"; String s1(s); String s2("123456"); s1.ShowStr(); s2.ShowStr(); 运行结果: string:ABCDE, length:5 string:123456, length:6
28
7.2.3 拷贝构造函数 拷贝构造函数是与类名相同,形参是本类的对象的引用的函数,在用已存在对象初始化新建立对象时调用。
C++语言程序设计教程 第7章 类与对象 7.2.3 拷贝构造函数 拷贝构造函数是与类名相同,形参是本类的对象的引用的函数,在用已存在对象初始化新建立对象时调用。 类的拷贝构造函数一般由用户定义,如果用户没有定义拷贝构造函数,系统就会自动生成一个默认函数,这个默认拷贝构造函数的功能是把初始值对象的每个数据成员的值依次复制到新建立的对象中。因此,也可以说是完成了同类对象的克隆(Clone)。这样得到的对象和原对象具有完全相同的数据成员,即完全相同的属性。
29
7.2.3 拷贝构造函数 定义一个拷贝构造函数的一般形式为: 类名(类名& 对象名)
C++语言程序设计教程 第7章 类与对象 7.2.3 拷贝构造函数 用户可以也可以根据实际问题的需要定义特定的拷贝构造函数来改变缺省拷贝构造函数的行为,以实现同类对象之间数据成员的传递。如果用户自定义了拷贝构造函数,则在用一个类的对象初始化该类的另外一个对象时,自动调用自定义的拷贝构造函数。 定义一个拷贝构造函数的一般形式为: 拷贝构造函数在用类的一个对象去初始化该类的另一个对象时调用,以下三种情况相当于用一个已存在的对象去初始化新建立的对象, 此时, 调用拷贝构造函数: ① 当用类的一个对象去初始化该类的另一个对象时。 ② 如果函数的形参是类的对象,调用函数时,将对象作为函数实参传递给函数的形参时。 ③ 如果函数的返回值是类的对象,函数执行完成,将返回值返回时。 类名(类名& 对象名) { … };
30
/***************************** * p7_4.cpp * * 构造拷贝构造函数 *
第7章 类与对象 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 /***************************** * p7_4.cpp * * 构造拷贝构造函数 * *****************************/ #include <iostream> using namespace std; class Clock { private: int H,M,S; public: Clock(int h=0,int m=0,int s=0) { H=h,M=m,S=s; cout<<"constructor:"<<H<<":"<<M<<":"<<S<<endl; } ~Clock() cout<<"destructor:"<<H<<":"<<M<<":"<<S<<endl;
31
注意: 运行结果: 拷贝构造函数只是在用一个已存在的对象去初始化新建立的对象时调用,在对 象进行赋值时,拷贝构造函数不被调用。
C++语言程序设计教程 第7章 类与对象 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 Clock(Clock & p) { cout<<"copy constructor,before call: "<<H<<":"<<M<<":"<<S<<endl; H=p.H; M=p.M; S=p.S; } void ShowTime() cout<<H<<":"<<M<<":"<<S<<endl; }; Clock fun(Clock C) return C; void main(void) Clock C1(8,0,0); Clock C2(9,0,0); Clock C3(C1); fun(C2); Clock C4; C4=C2; 运行结果: constructor:8:0:0 constructor:9:0:0 copy constructor,before call: : : copy constructor,before call: : : destructor:9:0:0 constructor:0:0:0 destructor:8:0:0 注意: 拷贝构造函数只是在用一个已存在的对象去初始化新建立的对象时调用,在对 象进行赋值时,拷贝构造函数不被调用。 用一个常量初始化新建立的对象时,调用构造函数,不调用拷贝构造函数。 建立对象时,构造函数与拷贝构造函数有且仅有一个被调用。
32
C++语言程序设计教程 第7章 类与对象 7.2.4 浅拷贝与深拷贝 在默认的拷贝构造函数中,拷贝的策略是直接将原对象的数据成员值依次拷贝给新对象中对应的数据成员,如前面示例p7_4.cpp中定义的拷贝函数所示,那么我们为何不直接使用系统默认的拷贝构造函数,何必又自己定义一个拷贝构造函数呢?但是,有些情况下使用默认的拷贝构造函数却会出现意想不到的问题。 例如,使用程序p7_3.cpp中定义的String类,执行下列程序系统就会出错: void main() { String s1("123456"); String s2=s1; }
33
C++语言程序设计教程 第7章 类与对象 7.2.4 浅拷贝与深拷贝 为什么会出错呢?程序中首先创建对象s1, 为对象s1分配相应的内存资源,调用构造函数初始化该对象,然后调用系统缺省的拷贝构造函数将对象s1拷贝给对象s2,这一切看来似乎很正常,但程序的运行却出现异常! 原因在于默认的拷贝构造函数实现的只能是浅拷贝,即直接将原对象的数据成员值依次拷贝给新对象中对应的数据成员,并没有为新对象另外分配内存资源。这样,如果对象的数据成员是指针,两个指针对象实际上指向的是同一块内存空间。 当执行String s2=s1时,默认的浅拷贝构造函数进行的是下列操作: s2.len=s1.len; s2.Str=s1.Str; 实际上是将s1.Str的地址赋给了s2.Str,并没有为s2.Str分配内存,执行String s2=s1;后,对象s2析构,释放内存,然后对象s1析构,由于s1.Str 和s2.Str所占用的是同一块内存,而同一块内存不可能释放两次,所以当对象s1析构时,程序出现异常,无法正常执行和结束。 由此可见,在某些情况下,浅拷贝会带来数据安全方面的隐患。
34
C++语言程序设计教程 第7章 类与对象 7.2.4 浅拷贝与深拷贝 当类的数据成员中有指针类型时,我们就必须定义一个特定的拷贝构造函数,该拷贝构造函数不仅可以实现原对象和新对象之间数据成员的拷贝,而且可以为新的对象分配单独的内存资源,这就是深拷贝构造函数。
35
7.2.4 浅拷贝与深拷贝 注意: 【例7-5】 带深拷贝构造函数的字符串类。
C++语言程序设计教程 第7章 构造数据类型 7.2.4 浅拷贝与深拷贝 【例7-5】 带深拷贝构造函数的字符串类。 在程序p7_3中的String类中,加入下列拷贝构造函数,构成了带深拷贝函数的字符串类。 String(String & r) { len=r.len; if(len!=0) Str=new char[len+1]; strcpy(Str,r.Str); } 下列程序能正常运行。 void main(void){ String s1("123456"); String s2=s1; s1.ShowStr(); s2.ShowStr(); 注意: 在重新定义拷贝构造函数后,默认拷贝构造函数与默认构造函数就不存在了, 如果在此时调用默认构造函数就会出错。 在重新定义构造函数后,默认构造函数就不存在了,但默认拷贝构造函数还存 在。 在对象进行赋值时,拷贝构造函数不被调用。此时进行的是结构式的拷贝。
36
C++语言程序设计教程 第7章 类与对象 7.3 对象的使用 类相当于一种包含函数的自定义数据类型,它不占内存,是一个抽象的“虚”体,使用已定义的类建立对象就像用数据类型定义变量一样。对象建立后,对象占据内存,变成了一个“实”体。 对象如同一般变量,占用一块连续的内存区域,因此可以使用一个指向对象的指针来访问对象,即对象指针,它指向存放该对象的地址。可用类来定义对象指针变量,通过对象指针来访问对象的成员。
37
7.3.1 对象指针 对象指针遵循一般变量指针的各种规则,其语法定义形式如下: 类名 *对象指针名;
C++语言程序设计教程 第7章 类与对象 7.3.1 对象指针 对象指针遵循一般变量指针的各种规则,其语法定义形式如下: 如同通过对象名访问对象的成员一样,使用对象指针也只能访问该类的公有数据成员和函数成员,但与前者使用“.”运算符不同,对象指针采用“->”运算符访问公有数对象指针名->数据成员名 例如: Clock *Cp; Clock C1(8,0,0); Clock *Cp; Cp=&C1; Cp->ShowTime(); 类名 *对象指针名; 对象指针名->数据成员名 或: 对象指针名->成员函数名(参数表)
38
C++语言程序设计教程 第7章 类与对象 7.3.1 对象指针 在C++中,对象指针可以作为成员函数的形参,一般而言,使用对象指针作为函数的参数要比使用对象作为函数的参数更普遍一些,因为使用对象指针作为函数的参数有如下两点好处: (1) 实现地址传递。 通过在函数调用时将实参对象的地址传递给形参指针对象,使形参指针对象和实参对象指向同一内存地址,这样,对象指针所指向对象的改变也将同样影响着实参对象,从而实现信息的双向传递。 (2) 使用对象指针效率高 使用对象指针传递的仅仅是对应实参对象的地址,并不需要实现对象之间的副本拷贝,这样就会减小时空开销,提高运行效率。
39
【例7-6】 时间加法。时间加法有两种, 一种是时钟加秒数, 另一种是时钟加时、分、秒。采用重载函数实现这两种加法。 1 2 3 4 5 6
C++语言程序设计教程 第7章 类与对象 【例7-6】 时间加法。时间加法有两种, 一种是时钟加秒数, 另一种是时钟加时、分、秒。采用重载函数实现这两种加法。 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 /***************************** * p7_6.cpp * * 带时间加法的时钟类 * *****************************/ #include <iostream> using namespace std; class Clock { private: int H,M,S; public: void SetTime(int h,int m,int s) { H=h,M=m,S=s; } void ShowTime() cout<<H<<":"<<M<<":"<<S<<endl;
40
C++语言程序设计教程 第7章 类与对象 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 Clock(int h=0,int m=0,int s=0) { H=h,M=m,S=s; } Clock(Clock & p) H=p.H, M=p.M, S=p.S; void TimeAdd(Clock *Cp); void TimeAdd(int h,int m,int s); void TimeAdd(int s); }; void Clock::TimeAdd(Clock *Cp) H=(Cp->H+H+(Cp->M+M+(Cp->S+S)/60)/60)%24; M=(Cp->M+M+(Cp->S+S)/60)%60; S=(Cp->S+S)%60; 函数重载 函数原型声明 对象指针
41
运行结果: 1:6:40 9:27:0 函数重载 函数重载 C++语言程序设计教程 第7章 类与对象 37 38 39 40 41 42
43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 void Clock::TimeAdd(int h,int m,int s) { H=(h+H+(m+M+(s+S)/60)/60)%24; M=(m+M+(s+S)/60)%60; S=(s+S)%60; } void Clock::TimeAdd(int s) H=(H+(M+(S+s)/60)/60)%24; M=(M+(S+s)/60)%60; S=(S+s)%60; void main() Clock C1; Clock C2(8,20,20); C1.TimeAdd(4000); C1.ShowTime(); C2.TimeAdd(&C1); C2.ShowTime(); 函数重载 函数重载 运行结果: 1:6:40 9:27:0
42
C++语言程序设计教程 第7章 类与对象 7.3.2 对象引用 对象引用就是对某类对象定义一个引用,其实质是通过将被引用对象的地址赋给引用对象,使二者指向同一内存空间,这样引用对象就成为了被引用对象的“别名”。 对象引用的定义方法与基本数据类型变量引用的定义是一样的。定义一个对象引用,并同时指向一个对象的格式为: 类名 & 对象引用名=被引用对象; 注意: 对象引用与被引用对象必须是同类型的。 除非是作为函数参数与函数返回值,对象引用在定义时必须要初始化。 定义一个对象引用并没有定义一个对象,所以不分配任何内存空间,不 调用构造函数。
43
7.3.2 对象引用 对象引用的使用格式为: 例如: Clock C1(8,20,20);
第7章 类与对象 7.3.2 对象引用 对象引用的使用格式为: 例如: Clock C1(8,20,20); Clock & Cr=C1; //定义了C1的对象引用Cr。 Cr.ShowTime(); //通过对象引用使用对象的成员 运行结果为: 8:20:20 对象引用名 . 数据成员名 或: 对象引用名 . 成员函数名(参数表)
44
C++语言程序设计教程 第7章 类与对象 7.3.2 对象引用 对象引用的优点: 对象引用通常用作函数的参数,它不仅具有对象指针的优点,而且比对象指针更简洁,更方便,更直观。将p7_6.cpp中添加如下函数: void Clock::TimeAdd(Clock & Cr) { H=(Cr.H+H+(Cr.M+M+(Cr.S+S)/60)/60)%24; M=(Cr.M+M+(Cr.S+S)/60)%60; S=(Cr.S+S)%60; } 将C2.TimeAdd(&C1); 替换为:C2.TimeAdd(C1); 运行结果与p7_6.cpp一样。
45
7.3.3 对象数组 对象数组是以对象为元素的数组。对象数组的定义、赋值、引用与普通数组一样,只是数组元素与普通数组的数组元素不同。
C++语言程序设计教程 第7章 类与对象 7.3.3 对象数组 对象数组是以对象为元素的数组。对象数组的定义、赋值、引用与普通数组一样,只是数组元素与普通数组的数组元素不同。 对象数组定义格式如下: 其中,类名指出该数组元素所属的类,常量表达式给出某一维元素的个数。 与结构数组不同,对象数组初始化需要使用构造函数完成,以一个大小为n的一维数组为例,对象数组的初始化格式如下: 类名 对象数组名[常量表达式n],...,[ 常量表达式2][常量表达式1]; 数组名[n]={ 构造函数(数据成员1初值,数据成员2初值,…), 构造函数(数据成员1初值,数据成员2初值,…), … 构造函数(数据成员1初值,数据成员2初值,…)}; 注意:不带初始化表的对象数组,其初始化靠调用不带参数的构造函数完成。
46
对象数组名[下标表达式1][ 下标表达式2]…[下标表达式m].成员函数名(参数表)
C++语言程序设计教程 第7章 类与对象 7.3.3 对象数组 以一个m维数组为例,对象数组元素的存取格式如下: 对象数组名[下标表达式1][ 下标表达式2]…[下标表达式m].数据成员名 或: 对象数组名[下标表达式1][ 下标表达式2]…[下标表达式m].成员函数名(参数表) 【例7-7】 计算一个班学生某门功课的总评成绩。 分析: 首先设计一个类Score,这个类的数据成员为一个学生的学号、姓名、平时成绩、期末考试成绩,成员函数有求总评成绩、显示成绩。然后,定义一个对象数组存储一个班学生的成绩。最后,通过逐一调用数组元素的成员函数求每个学生的总评成绩。
47
/***************************************** * p7_7.cpp *
第7章 类与对象 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 /***************************************** * p7_7.cpp * * 求一个班学生某门功课的总评成绩 * ******************************************/ #include<iostream> using namespace std; const int MaxN=100; const double Rate=0.6; //平时成绩比例 class Score { private: long No; //学号 char *Name; //姓名 int Peace; //平时成绩 int Final; //期末考试成绩 int Total; //总评成绩 public: Score(long=0,char* = NULL,int=0,int=0,int=0); //构造函数 void Count(); //计算总评成绩 void ShowScore(); //显示成绩 };
48
Score::Score(long no,char *name,int peace,int final,int total) //构造函数
第7章 类与对象 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 Score::Score(long no,char *name,int peace,int final,int total) //构造函数 { No=no; Name=name; Peace=peace; Final=final; Total=total; } void Score::Count() Total=Peace*Rate+Final*(1-Rate)+0.5; void Score::ShowScore() cout<<No<<"\t"<<Name<<"\t"<<Peace<<"\t"<<Final<<"\t"<<Total<<endl;
49
C++语言程序设计教程 第7章 类与对象 38 39 40 41 42 43 44 45 46 47 48 49 50 void main() { Score ClassScore1[3]; Score ClassScore2[3]={Score( ,"Tom",80,79), Score( ,"John",90,85), Score( ,"Wilson",70,55)}; for(int i=0;i<3;i++) ClassScore2[i].Count(); for(i=0;i<3;i++) ClassScore2[i].ShowScore(); } 对象数组 运行结果: Tom John Wilson
50
7.3.4 动态对象 动态对象: 动态对象是指编程者随时动态建立并可随时消失的对象。
C++语言程序设计教程 第7章 类与对象 7.3.4 动态对象 动态对象: 动态对象是指编程者随时动态建立并可随时消失的对象。 建立动态对象采用动态申请内存的语句new,删除动态对象使用delete语句。 建立一个动态对象的格式为: 对象指针=new 类名(初值表); 注意: 对象指针的类型应与类名一致。 动态对象存储在new语句从堆申请的空间中。 建立动态对象时要调用构造函数,当初值表缺省时调用默认的构造函数。
51
7.3.4 动态对象 例如: Clock *Cp; //建立对象指针
第7章 类与对象 7.3.4 动态对象 例如: Clock *Cp; //建立对象指针 Cp=new Clock; //建立动态对象,调用默认构造函数Clock()。 Cp->ShowTime(); //结果为0:0:0 Cp=new Clock(8,0,0); //建立动态对象,调用构造函数Clock(int,int,int) Cp->ShowTime(); //结果为8:0:0 注意: 函数体内的局部对象在函数调用时建立,在函数调用完后消失;全局对象则在程序执行时建立,执行完成后才消失;这些对象在何时建立,何时消失是C++规定好了的,不是编程者能控制的。
52
7.3.4 动态对象 在堆中建立的动态对象不能自动消失,需要使用delete语句删除对象,格式为: delete 对象指针;
C++语言程序设计教程 第7章 类与对象 7.3.4 动态对象 在堆中建立的动态对象不能自动消失,需要使用delete语句删除对象,格式为: 在删除动态对象时,释放堆中的内存空间,在对象消失时,调用析构函数。 例如: delete Cp; //删除Cp指向的动态对象 动态对象的一个重要的使用方面是用动态对象组成动态对象数组,建立一个一维动态对象数组的格式为: 删除一个动态对象数组的格式为: 在建立动态对象数组时,要调用构造函数,调用的次数与数组的大小相同;删除对象数组时,要调用析构函数,调用次数与数组的大小相同。 delete 对象指针; 对象指针 = new 类名[数组大小]; delete [] 对象指针;
53
7.3.4 动态对象 将p7_7.cpp改为用动态对象数组实现如下: { No=no; Name=name; Peace=peace;
第7章 类与对象 7.3.4 动态对象 将p7_7.cpp改为用动态对象数组实现如下: Score::SetScore(long no,char *name,int peace,int final,int total) { No=no; Name=name; Peace=peace; Final=final; Total=total; }
54
7.3.4 动态对象 SetScore()函数为动态数组设置初值。 void main() {
第7章 类与对象 7.3.4 动态对象 将p7_7.cpp改为用动态对象数组实现如下: SetScore()函数为动态数组设置初值。 void main() { Score * ClassScore; ClassScore=new Score [3]; ClassScore[0].SetScore( ,"Tom",80,79), ClassScore[1].SetScore( ,"John",90,85), ClassScore[2].SetScore( ,"Wilson",70,55); for(int i=0;i<3;i++) ClassScore[i].Count(); for(i=0;i<3;i++) ClassScore[i].ShowScore(); delete [] ClassScore; }
55
7.3.5 This指针 例如: void Clock::SetTime (int h, int m, int s) 1 2 {
第7章 类与对象 7.3.5 This指针 一个类的成员函数中,有时希望引用调用的它对象,对此,C++采用隐含的this指针来实现。 this指针是一个系统预定义的特殊指针,指向当前对象,表示当前对象的地址。 为了与类的数据成员H、M、S区别,将SetTime的形参名设为h、m、s。如果使用this指针,就可以凭this指针区分本对象的数据成员与其他变量。 1 2 3 4 5 6 例如: void Clock::SetTime (int h, int m, int s) { H=h, M=m, S=s; this->H=h, this->M=m, this->S=s; (* this).H=h, (* this).M=m, (* this).S=s; } // 语句3、4、5是等效的
56
C++语言程序设计教程 第7章 类与对象 7.3.5 This指针 系统利用this指针明确指出成员函数当前操作的数据成员所属的对象。实际上,当一个对象调用其成员函数时,编译器先将该对象的地址赋给this指针,然后调用成员函数,这样成员函数对对象的数据成员进行操作时,就隐含使用了this指针。 一般而言,通常不直接使用this指针来引用对象成员,但在某些少数情况下,可以使用this指针,如:重载某些运算符以实现对象的连续赋值等。 1 2 3 4 //使用this指针重新设计的SetTime()成员函数如下: void Clock::SetTime (int H, int M, int S) { this->H=H, this->M=M, this->S=S; } 注意: this指针不是调用对象的名称,而是指向调用对象的指针的名称。 this的值不能改变,它总是指向当前调用对象。
57
7.3.6 对象组合 组合概念体现的是一种包含与被包含的关系,在语义上表现为“is part of”的关系,即在逻辑上A是B的一部分 。
C++语言程序设计教程 第7章 类与对象 7.3.6 对象组合 组合概念体现的是一种包含与被包含的关系,在语义上表现为“is part of”的关系,即在逻辑上A是B的一部分 。 在C++程序设计中,类的组合用来描述一类复杂的对象,在类的定义中,它的某些属性,往往是另一个类的对象,而不是像整型、浮点型之类的简单数据类型,也就是“一个类内嵌其它类的对象作为成员”,将对象嵌入到类中的这样一种描述复杂类的方法,我们称之为“类的组合”,一个含有其他类对象的类称为组合类,组合类的对象称为组合对象。 组合类定义的步骤为:先定义成员类,再定义组合类。
58
7.3.6 对象组合 【例7-8】 计算某次火车的旅途时间。
C++语言程序设计教程 第7章 类与对象 7.3.6 对象组合 【例7-8】 计算某次火车的旅途时间。 分析:某次火车有车次、起点站、终点站、出发时间、到达时间。前面定义的Clock类正具有时间特性,因此,可以利用Clock对象组合成一个火车旅途类TrainTrip。假定火车均为24小时内到达,旅途时间为到达时间减出发时间。 用空方框表示类,灰框表示对象,组合类可以表示为空框包含灰框。设计TrainTrip类的示意图与成员构成图如图7-4:
59
Clock类 C++语言程序设计教程 第7章 类与对象 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17
18 19 20 /***************************** * p7_8.cpp * * 计算火车旅途时间的组合类 * *****************************/ #include <iostream> using namespace std; class Clock { private: int H,M,S; public: void ShowTime() { cout<<H<<":"<<M<<":"<<S<<endl; } void SetTime(int H=0,int M=0,int S=0) { this->H=H, this->M=M, this->S=S; } Clock(int H=0,int M=0,int S=0) Clock类
60
Clock StartTime; //出发时间 Clock EndTime; //到达时间 public:
第7章 类与对象 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 int GetH() { return H; } int GetM() { return M; } int GetS() { return S; } }; class TrainTrip { private: char *TrainNo; //车次 Clock StartTime; //出发时间 Clock EndTime; //到达时间 public: TrainTrip(char * TrainNo, Clock S, Clock E) { this->TrainNo=TrainNo; StartTime=S; EndTime=E; } 组合类
61
21:50:52 运行结果: 组合对象及其初始化 C++语言程序设计教程 第7章 类与对象 41 42 43 44 45 46 47 48
49 50 51 52 53 54 55 56 57 58 59 60 Clock TripTime() { int tH,tM,tS; //临时存储小时、分、秒数 int carry=0; //借位 Clock tTime; //临时存储时间 (tS=EndTime.GetS()-StartTime.GetS())>0?carry=0:tS+=60,carry=1; (tM=EndTime.GetM()-StartTime.GetM()-carry)>0?carry=0:tM+=60,carry=1; (tH=EndTime.GetH()-StartTime.GetH()-carry)>0?carry=0:tH+=24; tTime.SetTime(tH,tM,tS); return tTime; } }; void main() Clock C1(8,10,10), C2(6,1,2); //定义Clock类的对象 Clock C3; //定义Clock类对象,存储结果 TrainTrip T1("K16",C1,C2); //定义TrainTrip对象 C3=T1.TripTime(); C3.ShowTime(); 运行结果: 21:50:52 组合对象及其初始化
62
7.3.6 对象组合 C++为组合对象提供了初始化机制:在定义组合类的构造函数,可以携带初始化表,其格式如下
第7章 类与对象 7.3.6 对象组合 C++为组合对象提供了初始化机制:在定义组合类的构造函数,可以携带初始化表,其格式如下 组合类名(形参表):成员对象1(子形参表1),成员对象2(子形参表2),… 成员对象初始化表 建立对象时,调用组合类的构造函数;在调用组合类的构造函数时,先调用各个成员对象的构造函数,成员对象的初值从初始化列表中取得。这样,实际上是通过成员类的构造函数对成员对象进行初始化,初始化值在初始化表中提供。 为组合类定义了带初始化表的构造函数后,在建立组合对象同时为对象提供初值的格式如下: 类名 对象名(实参表);
63
C++语言程序设计教程 第7章 类与对象 7.3.6 对象组合 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 //使用初始化列表,将p7_8.cpp修改成p7_8a.cpp其中的构造函数修改如下: TrainTrip(char * TrainNo, int SH,int SM, int SS, int EH, int EM,int ES): EndTime(EH,EM,ES), StartTime(SH,SM,SS) { this->TrainNo=TrainNo; } //定义组合对象的程序修改如下: void main() Clock C3; //定义Clock类对象,存储结果 TrainTrip T1("K16",8,10,10,6,1,2); //定义TrainTrip对象 C3=T1.TripTime(); C3.ShowTime(); 成员对象初始化表 注意: 初始化列表既不能决定是否调用成员对象的构造函数,也不能决定调用构造函数的顺序,成员对象调用顺序由成员对象定义的顺序决定。初始化列表只是提供调用成员对象构造函数的参数
64
C++语言程序设计教程 第7章 类与对象 7.3.6 对象组合 组合对象的构造函数: 在定义一个组合类的对象时,不仅它自身的构造函数将被调用,而且还将调用其成员对象的构造函数,调用先后顺序为: ① 成员对象按照在其组合类的声明中出现的次序,依次调用各自的构造函数, 而不是按初始化列表中的顺序。如果建立组合类的成员对象时没有指定对象的初始值,则自动调用默认的构造函数。 ② 组合类对象调用组合类构造函数。 ③ 调用析构函数,析构函数的调用顺序与构造函数正好相反。
65
C++语言程序设计教程 第7章 类与对象 7.4 静态成员 一个类对象的public成员可被本类的其他对象存取,即可供所有对象使用,但是此类的每个对象都各自拥有一份,不存在真正意义上的共享成员。所以,C++提供了静态成员,用以解决同一个类的不同对象之间数据成员和函数的共享问题。 静态成员的特点是:不管这个类创建了多少个对象,其静态成员在内存中只保留一份拷贝,这个拷贝为该类的所有对象所共享。 类的静态成员有两种:静态数据成员和静态函数成员,下面分别对它们进行讨论
66
C++语言程序设计教程 第7章 类与对象 静态数据成员 静态数据成员: 是类的数据成员的一种特例,采用static关键字来定义,属于类属性,每个类只有一个拷贝,由该类的所有对象共同维护和使用,从而实现了同类的不同对象之间的数据共享。 面向对象方法中还有类属性(class attribute)的概念,类属性是描述类的所有对象的共同特征的一个数据项,对于任何对象实例,它的属性值是相同的,C++通过静态数据成员来实现类属性。
67
7.4.1 静态数据成员 静态数据成员的定义分为两个必不可少的部分:类内声明、类外初始化。 在类内,声明静态数据成员的格式为 :
C++语言程序设计教程 第7章 类与对象 静态数据成员 静态数据成员的定义分为两个必不可少的部分:类内声明、类外初始化。 在类内,声明静态数据成员的格式为 : static 数据类型 静态数据成员名; 在类外初始化的形式为: 数据类型 类名::静态数据成员名=初始值; 除了在初始化时可以在类外通过类对静态成员赋初值外,其他情况下对静态成员的存取规则与一般成员相同。即:在类内可以任意存取;在类外,通过对象只能访问存取属性为public的成员。
68
7.4.1 静态数据成员 对静态数据成员的定义与使用应注意 :
C++语言程序设计教程 第7章 类与对象 静态数据成员 对静态数据成员的定义与使用应注意 : 静态数据成员的访问属性同普通数据成员一样,可以为public、private和 protected。 静态数据成员脱离具体对象而独立存在,其存储空间是独立分配的,不是 任何对象存储空间的一部分,但逻辑上所有的对象都共享这一存储单元, 所以对静态数据成员的任何操作都将影响共享这一存储单元的所有对象。 静态数据成员是一种特殊的数据成员,它表示类属性,而不是某个对象单 独的属性,它在程序开始产生,在程序结束时消失。静态数据成员具有静 态生存期。 由于在类的定义中仅仅是对静态数据成员进行了引用性声明,因此必须在 文件作用域的某个地方对静态数据成员进行定义并初始化,即应在类体外 对静态数据成员进行初始化(静态数据成员的初始化与它的访问控制权限 无关)。 静态数据成员初始化时前面不加static关键字,以免与一般静态变量或对 象混淆。 由于静态数据成员是类的成员,因此在初始化时必须使用类作用域运算 符::限定它所属的类。
69
C++语言程序设计教程 第7章 类与对象 静态数据成员 例如: 我们对某学校不同学生的特性进行抽象,找出共性设计一个学生类Student,如果需要统计学生的总人数,可以在类外定义一个全局变量。但是类外的全局变量不受类存取控制的保护。因此可以将学生人数定义为静态成员,即学生类的类属性, 设计的Student类如下 : 1 2 3 4 5 6 7 8 class Student { private: char *Name; int No; static int countS; }; //在类外对静态成员初始化如下: int Student::countS=0; 注意: 对静态数据成员的访问还可以通过类的成员函数来进行。
70
C++语言程序设计教程 第7章 类与对象 静态数据成员 静态数据成员: 是类的数据成员的一种特例,采用static关键字来定义,属于类属性,每个类只有一个拷贝,由该类的所有对象共同维护和使用,从而实现了同类的不同对象之间的数据共享。 面向对象方法中还有类属性(class attribute)的概念,类属性是描述类的所有对象的共同特征的一个数据项,对于任何对象实例,它的属性值是相同的,C++通过静态数据成员来实现类属性。
71
C++语言程序设计教程 第7章 类与对象 静态成员函数 静态数据成员为类属性,在定义类后、建立对象前就存在。因此,在建立对象前不能通过成员函数存取静态数据成员。C++提供了静态成员函数,用来存取类的静态成员。 静态成员函数是用关键字static声明的成员函数,它属于整个类而不属于类中的某个对象,是该类的所有对象共享的成员函数。 静态成员函数可以在类体内定义。也可以在类内声明为static,在类外定义。当在类外定义时,不能再使用static关键字作为前缀。 静态函数成员的调用形式有如下两种: (1) 通过类名调用静态成员函数; (2) 通过对象调用静态成员函数
72
7.4.1 静态数据成员 (1) 通过类名调用静态成员函数
C++语言程序设计教程 第7章 类与对象 静态数据成员 (1) 通过类名调用静态成员函数 静态函数成员为类的全体对象而不是部分对象服务,与类相联系而不与类的对象联系,因此访问静态函数成员时,可以直接使用类名。格式如下: 类名::静态成员函数; 注意: 通过类名访问静态成员函数受静态成员函数访问权限的控制。 (2) 通过对象调用静态成员函数,格式为 : 对象. 静态成员函数
73
7.4.2 静态成员函数 通过对象调用静态成员函数应注意: 通过对象访问静态成员函数的前提条件为对象已经建立。
C++语言程序设计教程 第7章 类与对象 静态成员函数 通过对象调用静态成员函数应注意: 通过对象访问静态成员函数的前提条件为对象已经建立。 静态成员函数的访问权限同普通成员函数一样。 静态成员函数也可以省略参数、使用默认形参值、以及进行重载。 静态成员函数与普通成员函数在使用还有如下区别: 由于静态成员函数在类中只有一个拷贝(副本),因此它访问对象的成 员时要受到一些限制:静态成员函数可以直接访问类中说明的静态成 员,但不能直接访问类中的非静态成员;若要访问非静态成员时,必 须通过参数传递的方式得到相应的对象,再通过对象来访问。 由于静态成员是独立于类对象而存在的,因此静态成员没有this指 针。
74
静态数据成员 【例7-9】 使用静态成员维护内存中Student类对象的个数(对象计数器)。
C++语言程序设计教程 第7章 类与对象 【例7-9】 使用静态成员维护内存中Student类对象的个数(对象计数器)。 分析:为了维护内存中Student类对象的个数,除了定义一个静态数据成员存储类对象个数外,还要在所有可能建立对象、删除对象的场合记载对对象个数的修改。 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 /***************************** * p7_9.cpp * * 含有对象计数器的学生类 * ******************************/ #include<iostream> using namespace std; class Student { private: char *Name; int No; static int countS; public: static int GetCount() { return countS; } 静态数据成员 静态成员函数
75
静态数据成员 C++语言程序设计教程 第7章 类与对象 17 18 19 20 21 22 23 24 25 26 27 28 29 30
31 32 33 34 35 36 Student(char* ="", int=0); Student(Student &); ~Student(); }; Student::Student(char * Name, int No) { this->Name=new char [strlen(Name)+1]; strcpy(this->Name, Name); this->No=No; ++countS; cout<<"constructing:"<<Name<<endl; } Student::Student(Student & r) Name=new char [strlen(r.Name)+1]; strcpy(Name, r.Name); No=r.No; cout<<"copy constructing:"<<r.Name<<endl; 静态数据成员
76
运行结果: constructing:Antony 1 copy constructing:Antony 2 constructing: 4
第7章 类与对象 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 Student::~Student() { cout<<"destructing:"<<Name<<endl; delete [] Name; --countS; } int Student::countS=0; void main() cout<<Student::GetCount()<<endl; //使用类调用静态成员函数 Student s1("Antony"); //建立一个新对象 cout<<s1.GetCount()<<endl; //通过对象调用静态成员函数 Student s2(s1); //利用已有对象建立一个新对象 cout<<s1.GetCount()<<endl; Student S3[2]; //建立一个对象数组 cout<<Student::GetCount()<<endl; Student *s4=new Student[3]; //建立一动态对象数组 delete [] s4; //删除动态对象数组 运行结果: constructing:Antony 1 copy constructing:Antony 2 constructing: 4 7 destructing: destructing:Antony
77
C++语言程序设计教程 第7章 类与对象 7.5 友元 数据隐藏给不同类和对象的成员函数之间,类的成员函数和类外的一般函数之间进行属性共享带来障碍,必须寻求一种方法使得类外的对象能够访问类中的私有成员,提高程序的效率。 为了解决这个问题,C++提出了使用友元作为实现这一要求的辅助手段。 友元不是类的成员,但它可以访问类的任何成员(包括私有成员)。声明为友元的外界对象既可以是另一个类的成员函数,也可以是不属于任何类的一般的函数,称之为友元函数;友元也可以是整个的一个类,称之为友元类。
78
7.5.1 友元函数 friend 返回类型 函数名(形参表) { … //函数体 }
C++语言程序设计教程 第7章 类与对象 友元函数 友元函数是在类定义中由关键字friend修饰的非成员函数。其格式为 : friend 返回类型 函数名(形参表) { … //函数体 } 注意: 友元函数的定义与成员函数一样,只是在类中用关键字friend予以说明。但友 元函数是一个普通的函数,它不是本类的成员函数,因此在调用时不能通过对 象调用。 友元函数也可以在类内声明,在类外定义。 友元函数对类成员的存取与成员函数一样,可以直接存取类的任何存取控制属 性的成员;可通过对象存取形参、函数体中该类类型对象的所有成员。 private、protected、public访问权限与友员函数的声明无关,因此原则上, 友元函数声明可以放在类体中任意部分,但为程序清晰,一般放在类体的后 面。
79
【例7-10】 使用友元函数计算某次火车的旅途时间。
C++语言程序设计教程 第7章 类与对象 【例7-10】 使用友元函数计算某次火车的旅途时间。 分析:为了简化问题,分别用两个Clock类对象表示某次火车出发时间、到达时间。假定火车均为24小时内到达,旅途时间为到达时间减出发时间。 1 2 3 4 5 6 7 8 9 10 11 12 13 14 /****************************** * p7_10.cpp * * 计算火车旅途时间的友元函数 * *******************************/ #include <iostream> using namespace std; class Clock { private: int H,M,S; public: void ShowTime() { cout<<H<<":"<<M<<":"<<S<<endl; } void SetTime(int H=0,int M=0,int S=0) { this->H=H, this->M=M, this->S=S; }
80
C++语言程序设计教程 第7章 类与对象 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 Clock(int H=0,int M=0,int S=0) { this->H=H, this->M=M, this->S=S; } friend Clock TripTime(Clock & StartTime, Clock & EndTime); }; Clock TripTime(Clock & StartTime, Clock & EndTime) int tH,tM,tS; //临时存储小时、分、秒数 int carry=0; //借位 Clock tTime; //临时存储时间 (tS=EndTime.S-StartTime.S)>0?carry=0:tS=+60,carry=1; (tM=EndTime.M-StartTime.M-carry)>0?carry=0:tM+=60,carry=1; (tH=EndTime.H-StartTime.H-carry)>0?carry=0:tH+=24; tTime.SetTime(tH,tM,tS); return tTime; 友元函数
81
C++语言程序设计教程 第7章 类与对象 34 35 36 37 38 39 40 void main() { Clock C1(8,10,10), C2(6,1,2); //定义Clock类的对象 Clock C3; //定义Clock类对象,存储结果 C3=TripTime(C1,C2); C3.ShowTime(); } 运行结果: 21:50:52 说明: 在本例中,在Clock类体中设计了一个友元函数TripTime(), 它不是类的成员函 数。但是,可以看到友元函数中通过对象名StartTime和EndTime直接访问了它 们的的私有数据 成员StartTime.H、StartTime.M、StartTime.S,这就是友元的关键所在。 使用友元成员的好处是两个类可以某种方式相互合作、协同工作,共同完成某 一任务。
82
7.5.2 友元类 友元除可以是函数外,还可以是类,如果一个类声明为另一个类的友元,则该类称为另一个类的友元类。
C++语言程序设计教程 第7章 类与对象 友元类 友元除可以是函数外,还可以是类,如果一个类声明为另一个类的友元,则该类称为另一个类的友元类。 若A类为B类的友元类,则A类的所有成员函数都是B类的友元函数,都可以访问B类的任何数据成员。 友元类的声明是在类名之前加上关键字friend来实现。声明A类为B类的友员类的格式如下: class B { … friend class A; }; 注意: 在声明A类为B类的友元类时,A类必须已经存在,但是如果A类又将B类声明为自己的友员类时,又会出现B类不存在的错误。
83
7.5.2 友元类 前向引用声明: 当遇到两个类相互引用的情况时,必然有一个类在定义之前就将被引用,怎么办呢?
C++语言程序设计教程 第7章 类与对象 友元类 前向引用声明: 当遇到两个类相互引用的情况时,必然有一个类在定义之前就将被引用,怎么办呢? 对此,C++专门规定了前向引用声明用以解决这类问题,前向引用声明是在引用未定义的类之前对该类进行声明,它只是为程序引入一个代表该类的标识符,类的具体定义可以在程序的其他地方进行。 1 2 3 4 5 6 7 8 9 10 //前向引用声明示例 class B; //前向引用声明 class A { //A类的定义 public: //外部接口 void funA(B b); //以B类对象b为形参的成员函数 }; class B { //B类的定义 void funB(A a); //以A类对象a为形参的成员函数
84
【例7-11】使用友元类计算某次火车的旅途时间。
C++语言程序设计教程 第7章 类与对象 【例7-11】使用友元类计算某次火车的旅途时间。 分析:在p7_8.cpp中,定义了一个组合类TrainTrip,组合了Clock类对象表示某次火车出发时间、到达时间。但是, TrainTrip中的成员函数无法直接存取出发时间、到达时间中的访问控制为private的H、M、S。如果将TrainTrip定义为Clock的友员类,则TrainTrip中的成员函数可以直接存取出发时间、到达时间中的数据成员。 1 2 3 4 5 6 7 8 9 10 11 12 13 /***************************** * p7_11.cpp * * 计算火车旅途时间的友元类 * *****************************/ #include <iostream> using namespace std; class TrainTrip; //前向引用声明 class Clock { private: int H,M,S; public: void ShowTime() { cout<<H<<":"<<M<<":"<<S<<endl; }
85
C++语言程序设计教程 第7章 类与对象 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 void SetTime(int H=0,int M=0,int S=0) { this->H=H, this->M=M, this->S=S; } Clock(int H=0,int M=0,int S=0) friend class TrainTrip; }; class TrainTrip { private: char *TrainNo; //车次 Clock StartTime; //出发时间 Clock EndTime; //到达时间 public: TrainTrip(char * TrainNo, Clock S, Clock E) this->TrainNo=TrainNo; StartTime=S; EndTime=E;
86
运行结果: 21:50:52 C++语言程序设计教程 第7章 类与对象 36 37 38 39 40 41 42 43 44 45 46
47 48 49 50 51 52 53 54 55 56 Clock TripTime() { int tH,tM,tS; //临时存储小时、分、秒数 int carry=0; //借位 Clock tTime; //临时存储时间 (tS=EndTime.S-StartTime.S)>0?carry=0:tS+=60,carry=1; (tM=EndTime.M-StartTime.M-carry)>0?carry=0:tM+=60,carry=1; (tH=EndTime.H-StartTime.H-carry)>0?carry=0:tH+=24; tTime.SetTime(tH,tM,tS); return tTime; } }; void main() Clock C1(8,10,10), C2(6,1,2); //定义Clock类的对象 Clock C3; //定义Clock类对象,存储结果 TrainTrip T1("K16",C1,C2); //定义TrainTrip对象 C3=T1.TripTime(); C3.ShowTime(); 运行结果: 21:50:52
87
7.5.2 友元类 友元关系具有以下性质: 友元关系是不能传递的,B类是A类的友元,C类是B类的友元,C类和A
第7章 类与对象 友元类 友元关系具有以下性质: 友元关系是不能传递的,B类是A类的友元,C类是B类的友元,C类和A 类之间,如果没有声明,就没有任何友元关系,不能进行数据共享。 友元关系是单向的,如果声明B类是A类的友元,B类的成员函数就可 以访问A类的私有和保护数据,但A类的成员函数却不能访问B类的私 有和保护数据。 友元概念的引入,提高了数据的共享性,加强了函数与函数之间,类 与类之间的相互联系,大大提高程序的效率,这是友元的优点,但友 元也破坏了数据隐蔽和数据封装,导致程序的可维护性变差,给程序 的重用和扩充埋下了深深的隐患,这是友元的缺点。 对友元的使用必须慎重,要具体问题具体分析,在提高效率和增加共 享之间把握好一个“度”,在共享和封装之间进行恰当的折衷平衡。
88
C++语言程序设计教程 第7章 类与对象 7.6 常对象与常成员 由于常量是不可改变的,因此我们将“常”广泛用在C++中用来表示不可改变的量,如前面讲的常变量。不仅变量可以定义为常变量,对象、类的成员也可以定义为“常”。在程序中,我们有时候不允许修改某些特定的对象。如果某个对象不允许被修改,则该对象称为常对象。C++用关键字const来定义常对象。 const也可以用来限定类的数据成员和成员函数,分别称为类的常数据成员和常成员函数。C++中常对象、常数据成员、常成员函数的访问和调用各有其特别之处。 常对象和常成员概念的建立,明确规定了程序中各种对象的变与不变的界线,从而进一步增强了C++程序的安全性和可控性。
89
C++语言程序设计教程 第7章 类与对象 常对象 C++编译器对常对象(const对象)的使用是极为苛刻的,它不允许常对象调用任何类的成员函数,而且常对象一旦定义,在其生存期内不允许改变,否则将导致编译错误。 常对象定义格式如下: 类型 const 对象名; 或 const 类型 对象名; 1 2 3 4 5 6 7 8 9 10 void main( ) { const Clock C1(9,9,9); //定义常对象C1 Clock const C2(10,10,10); //定义常对象C2 Clock C3(11,11,11); // C1=C3; //错误!C1为常对象, 不能被赋值 // C1.ShowTime(); //错误!C1为常对象, 不能访问非常成员函数 C3. ShowTime(); // C1.SetTime(0,0,0); //错误!C1为常对象,不能被更新! }
90
C++语言程序设计教程 第7章 类与对象 常数据成员 使用const说明的数据成员称为常数据成员。常数据成员的定义与一般常变量的定义方式相同,只是它的定义必须出现在类体中。 常数据成员同样也必须进行初始化,并且不能被更新。但常数据成员的初始化只能通过构造函数的初始化列表进行。 常数据成员定义的格式如下 : 数据类型 const 数据成员名; 或 const 数据类型 数据成员名;
91
C++语言程序设计教程 第7章 类与对象 【例7-11】演示常数据成员的使用 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 #include<iostream> using namespace std; class A { private: const int& r; //常引用数据成员 const int a; //常数据成员 static const int b; //静态常数据成员 public: A(int i):a(i),r(a) //常数据成员只能通过初始化列表来获得初值 { cout<<"constructor!"<<endl; }; void display() cout<<a<<","<<b<<","<<r<<endl; }
92
运行结果: constructor! 1,3,1 2,3,2 void main() { A a1(1); a1.display();
第7章 类与对象 18 19 20 21 22 23 24 25 const int A::b=3; //静态常数据成员在类外说明和初始化 void main() { A a1(1); a1.display(); A a2(2); a2.display(); } 运行结果: constructor! 1,3,1 2,3,2
93
7.6.3 常成员函数 在定义时使用const关键字修饰的用于访问类的常对象的函数, 称为常成员函数。 常成员函数的说明格式如下:
第7章 类与对象 常成员函数 在定义时使用const关键字修饰的用于访问类的常对象的函数, 称为常成员函数。 常成员函数的说明格式如下: 返回类型 成员函数名 (参数表) const; 在定义与使用常成员函数时要注意: const是函数类型的一个组成部分,因此在函数实现部分也要带有 const关键字。 常成员函数不能更新对象的数据成员,也不能调用该类中没有用 const修饰的成员函数。 常对象只能调用它的常成员函数,而不能调用其他成员函数。这是 C++语法机制上对常对象的保护,也是常对象唯一的对外接口方式 。 const关键字可以用于参与重载函数的区分。
94
例如:我们可以定义一个日期类Date,通过常成员函数来读出年、月、日
C++语言程序设计教程 第7章 类与对象 常对象 例如:我们可以定义一个日期类Date,通过常成员函数来读出年、月、日 1 2 3 4 5 6 7 8 9 10 11 class Date { private: int Y, M, D; public: int year() const; int month() const; int day() const {return D; }; int day(){return D++;} int AddYear(int i) {return Y+i; }; };
95
C++语言程序设计教程 第7章 类与对象 常对象 有下列容易出现的错误: 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 ① int Date::month() //错误:常成员函数实现不能缺少const { return M; } ② int Date::year()const { // return Y++; //错误:常成员函数不能更新类的数据成员 return Y; } ③ Date const d1; // int j=d1.AddYear(10); //错误:常对象不能调用非常成员函数 int j= d1. year(); //正确 ④ Date d2; int i=d2.year(); //正确,非常对象可以调用常成员函数 d2.day(); //正确,非常对象可以调用非常成员函数
96
7.7 对象的内存分布 类只是一个型,除了静态数据成员外,在没有实例化成对象前不占任何内存。
C++语言程序设计教程 第7章 类与对象 7.7 对象的内存分布 类只是一个型,除了静态数据成员外,在没有实例化成对象前不占任何内存。 类的静态数据成员与全局对象(变量)一样,在数据段中分配内存。 对象的内存空间分配 当类被实例化成对象后,不同类别的对象占据不同类型的内存,其规律与普通变量相同: (1) 建立的全局对象占有数据段的内存。 (2) 建立的局部对象内存分配在栈中。 (3) 函数调用时为实参建立的临时对象内存分配在栈中。 (4) 使用动态内存分配语句new建立的动态对象,内存在堆中分配。
97
7.7 对象的内存分布 程序运行时,系统只为各对象的数据成员分配单独的内存空间,对象的内存空间分配有下列规则:
C++语言程序设计教程 第7章 类与对象 7.7 对象的内存分布 程序运行时,系统只为各对象的数据成员分配单独的内存空间,对象的内存空间分配有下列规则: ① 对象的数据成员与成员函数占据不同的内存空间,数据成员的内 存空间与对象的存储类别相关,成员函数的内存空间在代码段中。 ② 一个类所有对象的数据成员拥用各自的内存空间。 ③ 一个类所有对象的成员函数为该类的所有对象共享,在内存中,只 有一个拷贝。
98
7.7 对象的内存分布 2. 对象内存空间的释放 随着对象的生命周期的结束,对象所占的空间就会释放,各类对象内存空间释放时间与方法如下:
C++语言程序设计教程 第7章 类与对象 7.7 对象的内存分布 2. 对象内存空间的释放 随着对象的生命周期的结束,对象所占的空间就会释放,各类对象内存空间释放时间与方法如下: (1) 全局对象数据成员占有的内存空间在程序结束时释放。 (2) 局部对象与实参对象数据成员的内存空间在函数调用结束时释放 (3) 动态对象数据成员的内存空间要使用delete语句释放。 (4) 对象的成员函数的内存空间在该类的所有对象生命周期结束时自动释放。
99
本章小结 ◇在面向对象的程序设计中,程序模块是由类构成的。类是对逻辑上相关的函数与数 据的封装,它是对问题的抽象描述。
C++语言程序设计教程 第7章 类与对象 本章小结 ◇在面向对象的程序设计中,程序模块是由类构成的。类是对逻辑上相关的函数与数 据的封装,它是对问题的抽象描述。 ◇ 类中有数据成员与成员函数,成员的访问控制属性有private、protected、 public。类内可以访问所有控制属性的成员,在类外通过对象只能访问控制属性为 public的成员。 ◇ 类是“型”,是“虚”的,不占内存,使用类建立对象后,对象是“实”的,占有内存空 间。 ◇ 建立对象时调用构造函数初始化对象的数据成员,一个类提供默认的构造函数与默 认的拷贝构造函数。默认的构造函数是空的,默认的拷贝构造函数的内容为浅拷贝 语句。对象消失时会调用析构函数。构造函数与析构函数都可以重新定义。 ◇ 在默认的拷贝构造函数中,拷贝的策略是直接将原对象的数据成员值依次拷贝给新 对象中对应的数据成员,这种方式为浅拷贝。深拷贝能将原对象指针指向的内容 拷贝给新对象中。 ◇ 建立对象指针、对象引用均没有建立对象,所以此时不调用构造函数。通过对象指 针使用对象的成员要用操作符->, 通过对象引用使用对象的成员要用操作符. 。 ◇ 对象数组是以对象为元素的数组,对象数组的定义、赋值、引用与普通数组一样, 建立一个对象数组,相当于建立了多个对象,因此,多次调用构造函数。对象数组 初始化需要使用构造函数完成。。
100
本章小结 ◇建立动态对象使用语句new,动态对象一定要用语句delete删除。建立动态对象数组
C++语言程序设计教程 第7章 类与对象 本章小结 ◇建立动态对象使用语句new,动态对象一定要用语句delete删除。建立动态对象数组 使用语句new[],删除动态对象数组使用语句delete[]。 ◇ this指针是一个系统预定义的指向当前对象的指针,通过this指针可以访问对象的 成员。 ◇ 组合类是含有类对象的类,组合类对象称为组合对象。定义组合对象时调用构造函 数的顺序为类中成员对象定义的顺序,子对象构造时初始值通过组合类构造函数的 成员对象初始化表提供。 ◇ 静态数据成员是类的数据成员,独立于类存在。在类内定义,在类外初始化。静态 成员函数属于整个类,是该类的所有对象共享的成员函数。可通过类名、对象调用 静态成员函数访问静态函数成员。 ◇ 友元函数不是类的成员,但它可以访问类的任何成员。一个类的友元类可以访问该 类的任何成员。 ◇ 关键字const来定义的对象称为常对象,常对象的成员不允许被修改。使用const定 义的数据成员称为常数据成员,常数据成员不能被更新。在定义时使用const关键 字修饰的成员函数称为常成员函数,用于访问类的常对象。 ◇ 各类对象在内存中的分布与以及生命周期与普通变量一样变量一样,一个类的所有 对象共有该类的成员函数,独享各自的数据成
Similar presentations