1、第6章 面向对象的妥协本章学习要点本章学习要点友元友元友元函数友元函数友元类友元类静态成员静态成员静态数据成员静态数据成员静态成员函数静态成员函数第第6章章 面向对象的妥协面向对象的妥协本章学习目标本章学习目标了解友元函数的概念,掌握友元函数的定义与使用了解友元函数的概念,掌握友元函数的定义与使用了解友元类的概念了解友元类的概念掌握静态数据成员的定义与使用掌握静态数据成员的定义与使用掌握静态成员函数的定义与使用掌握静态成员函数的定义与使用第第6章章 面向对象的妥协面向对象的妥协v友元可以访问与其有好友关系的类中的任友元可以访问与其有好友关系的类中的任何成员。何成员。6.1 封装的破坏封装的破坏
2、友元友元 友元友元友元函数友元函数友元类友元类v如果在本类以外的其他地方定义了一个函数如果在本类以外的其他地方定义了一个函数(这个这个函数可以是不属于任何类的普通函数,也可以是函数可以是不属于任何类的普通函数,也可以是其他类的成员函数其他类的成员函数),在类体中用在类体中用friend对其进行对其进行声明声明,此函数就称为此函数就称为本类的友元函数本类的友元函数。友元函数。友元函数可以访问这个类中的任何成员。可以访问这个类中的任何成员。v如何将普通函数声明为友元函数呢?看下面这个如何将普通函数声明为友元函数呢?看下面这个简单的例子:简单的例子:6.1 封装的破坏封装的破坏友元友元 6.1.1
3、友元友元函数函数【例例6-1】友元普通函数。友元普通函数。#include using namespace std;class Clock /声明声明Clock类类 public:Clock(int,int,int);friend void display(Clock&);private:int hour;int minute;int second;6.1.1 友元友元 函数函数Clock:Clock(int h,int m,int s)/构造函数构造函数 hour=h;minute=m;second=s;void display(Clock&t)coutt.hour:t.minute:t.s
4、econdendl;int main()Clock t(10,13,56);display(t);/调用调用display函数,实参函数,实参t是是Clock类对象类对象 return 0;6.1 友元函数友元函数程序运行结果如下:程序运行结果如下:10:13:5610:13:56这是友元函数,这是友元函数,形参形参t是是Clock类类对象的引用对象的引用【例例6-2】友元成员函数。友元成员函数。#include using namespace std;class Date;class Clock /声明声明Clock类类public:Clock(int,int,int);void displ
5、ay(Date&);private:int hour,minute,second;6.1.1 友元友元 函数函数Clock类的类的成员函数成员函数对对Date类的提类的提前引用声明前引用声明class Date /声明声明Date类类public:Date(int,int,int);friend void Clock:display(Date&);private:int month;int day;int year;6.1.1 友元友元 函数函数Date类的类的友元函数友元函数Clock:Clock(int h,int m,int s)/Clock类的构造函数类的构造函数 hour=h;min
6、ute=m;second=s;void Clock:display(Date&dd)coutdd.month/dd.day/dd.yearendl;couthour:minute:secondendl;Date:Date(int m,int d,int y)/Date类的构造函数类的构造函数 month=m;day=d;year=y;6.1.1 友元函数友元函数int main()Clock t(10,13,56);/定义定义Clock类对象类对象t Date d(12,25,2004);/定义定义Date类对象类对象d /调用调用t的的display函数,实参是函数,实参是Date类对象类对
7、象d t.display(d);return 0;6.1 友元函数友元函数程序运行结果如下:程序运行结果如下:12/25/200412/25/200410:13:5610:13:56v不仅可以将一个函数声明为一个类的不仅可以将一个函数声明为一个类的“朋友朋友”,而且可以而且可以将一个类将一个类(例如例如B类类)声明为另一个类声明为另一个类(例例如如A类类)的的“朋友朋友”。这时。这时B类就是类就是A类的类的友元类友元类。友元类友元类B中的所有成员函数都是中的所有成员函数都是A类的友元函数,类的友元函数,可以访问可以访问A类中的任何成员。类中的任何成员。6.1 封装的破坏封装的破坏友元友元 6.
8、1.2 友元友元类类class Student;class Teacherpublic:void assigngrades(Student&s);/给出学生成绩给出学生成绩 protected:int stu_count;/学生人数学生人数 Student*plist100;/学生数组学生数组;6.1.2 友元类友元类class Studentpublic:friend Teacher;/友元类友元类 protected:int num;string name;float score;v声明友元类的一般形式为:声明友元类的一般形式为:friend 类名类名;14友元类举例友元类举例class
9、A friend class B;/B是是A的友元类的友元类 public:void Display()coutxendl;private:int x;class B public:void Set(int i);void Display();private:A a;15void B:Set(int i)a.x=i;void B:Display()a.Display();39友元类举例友元类举例6.1.2 友元类友元类v关于友元,有三点需要关于友元,有三点需要说明说明:v(1)友元函数的声明可以出现在类的任何友元函数的声明可以出现在类的任何地方地方(包括在(包括在private和和public
10、部分),也就部分),也就是说友元的说明不受成员访问控制符的限是说友元的说明不受成员访问控制符的限制。制。6.1.2 友元类友元类v(2)友元关系是单向的而不是双向的友元关系是单向的而不是双向的,如,如果声明了果声明了B类是类是A类的友元类,不等于类的友元类,不等于A类是类是B类的友元类,类的友元类,A类中的成员函数不一定能类中的成员函数不一定能够访问够访问B类中的成员类中的成员v(3)友元关系是不能传递的友元关系是不能传递的,例如,如果,例如,如果B类是类是A类的友元类,类的友元类,C类是类是B类的友元类,并类的友元类,并不能说不能说C类就是类就是A类的友元类。类的友元类。v在在C+中,声明了
11、一个类之后,可以定义该类中,声明了一个类之后,可以定义该类的多个对象。系统为每个对象分配单独的内存的多个对象。系统为每个对象分配单独的内存空间。每一个对象都分别有自己的数据成员,空间。每一个对象都分别有自己的数据成员,不同对象的数据成员各自有其值,互不相干。不同对象的数据成员各自有其值,互不相干。但是有时我们希望有某一个或几个数据成员为但是有时我们希望有某一个或几个数据成员为所有对象所共有。这样可以所有对象所共有。这样可以实现数据共享实现数据共享。6.2 对象机制的破坏对象机制的破坏 静态成员静态成员 v我们知道我们知道全局变量能够实现数据共享全局变量能够实现数据共享。但是用全局。但是用全局变
12、量的安全性得不到保证,因为在各处都可以自由变量的安全性得不到保证,因为在各处都可以自由地修改全局变量的值,很有可能由于某个没注意到地修改全局变量的值,很有可能由于某个没注意到的失误,全局变量的值就被修改,导致程序的失败的失误,全局变量的值就被修改,导致程序的失败。因此在实际工作中很少使用全局变量。因此在实际工作中很少使用全局变量。v如果想在同类的多个对象之间如果想在同类的多个对象之间实现数据共享实现数据共享,也不也不要用全局对象,可以用要用全局对象,可以用静态成员静态成员。静态成员包括静。静态成员包括静态数据成员和静态成员函数。态数据成员和静态成员函数。6.2 对象机制的破坏对象机制的破坏 静
13、态成员静态成员 v静态数据成员是一种特殊的数据成员静态数据成员是一种特殊的数据成员。它以关键字。它以关键字static开头。例如开头。例如:6.2 对象机制的破坏对象机制的破坏 静态成员静态成员 6.2.1 静态数据成员静态数据成员class Studentpublic:Student(char*name=no name);static int stu_count;char name40;v静态数据成员在内存中只占一份空间静态数据成员在内存中只占一份空间(而不是每个(而不是每个对象都分别为它保留一份空间),对象都分别为它保留一份空间),它是属于类的它是属于类的,但它被该类的所有对象所共享但它被
14、该类的所有对象所共享,每个对象都可以访,每个对象都可以访问这个静态数据成员。问这个静态数据成员。静态数据成员的值对所有对静态数据成员的值对所有对象都是一样的。象都是一样的。如果改变它的值,则在各对象中这如果改变它的值,则在各对象中这个数据成员的值都同时改变了。这样可以节约空间个数据成员的值都同时改变了。这样可以节约空间,提高效率。,提高效率。6.2.1 静态数据成员静态数据成员 v(1)如果只声明了类而未定义对象,则类的一般数据)如果只声明了类而未定义对象,则类的一般数据成员是不占内存空间的,只有在定义对象时,才为对象成员是不占内存空间的,只有在定义对象时,才为对象的数据成员分配空间。但是的数
15、据成员分配空间。但是静态数据成员不属于某一个静态数据成员不属于某一个对象,对象,在为对象所分配的空间中不包括静态数据成员所在为对象所分配的空间中不包括静态数据成员所占的空间占的空间。静态数据成员是在所有对象之外单独开辟空静态数据成员是在所有对象之外单独开辟空间间。只要在类中定义了静态数据成员,即使不定义对象只要在类中定义了静态数据成员,即使不定义对象,也为静态数据成员分配空间,它可以被访问。,也为静态数据成员分配空间,它可以被访问。在一个在一个类中可以有一个或多个静态数据成员,所有的对象共享类中可以有一个或多个静态数据成员,所有的对象共享这些静态数据成员,都可以访问它。这些静态数据成员,都可以
16、访问它。6.2.1 静态数据成员静态数据成员 说明:说明:v(2)静态数据成员不随对象的建立而分配空间,也静态数据成员不随对象的建立而分配空间,也不随对象的撤销而释放不随对象的撤销而释放(一般数据成员是在对象建立一般数据成员是在对象建立时分配空间,在对象撤销时释放时分配空间,在对象撤销时释放)。静态数据成员是。静态数据成员是在程序编译时被分配空间的,到程序结束时才释放在程序编译时被分配空间的,到程序结束时才释放空间。空间。6.2.1 静态数据成员静态数据成员 v(3)静态数据成员可以初始化,但只能在类体外进静态数据成员可以初始化,但只能在类体外进行初始化。行初始化。如:如:int Studen
17、t:stu_count=0;v静态数据成员可以初始化语句的一般形式为:静态数据成员可以初始化语句的一般形式为:数据类型数据类型 类名类名:静态数据成员名静态数据成员名=初值初值;v不必在初始化语句中加不必在初始化语句中加static。6.2.1 静态数据成员静态数据成员 v(4)静态数据成员既可以静态数据成员既可以通过对象名访问通过对象名访问,也可以,也可以通过类名来访问通过类名来访问。6.2.1 静态数据成员静态数据成员【例【例6-36-3】访问静态数据成员。访问静态数据成员。#include using namespace std;#include class Studentpublic:
18、Student(char*name=no name);Student();static int stu_count;char name40;6.2.1 静态数据成员静态数据成员 Student:Student(char*pname)strcpy(name,pname);stu_count+;/每创建一个对象,学生人数加每创建一个对象,学生人数加1Student:Student()stu_count-;/每释放一个对象,学生人数减每释放一个对象,学生人数减1/对静态数据成员对静态数据成员stu_count初始化初始化 int Student:stu_count=0;6.2.1 静态数据成员静态数
19、据成员 int main()Student s1;couts1.stu_countendl;Student s2;couts2.stu_count endl;coutStudent:stu_count endl;return 0;程序运行结果如下:程序运行结果如下:1 12 21 16.2.1 静态数据成员静态数据成员 v与静态数据成员不同,静态成员函数的作用与静态数据成员不同,静态成员函数的作用不是为不是为了对象之间的沟通,而是为了能处理静态数据成员了对象之间的沟通,而是为了能处理静态数据成员。6.2 对象机制的破坏对象机制的破坏 静态成员静态成员 6.2.2 静态成员静态成员函数函数【例【
20、例6-6-4 4】静态成员静态成员函数访问静态数据成员的例子。函数访问静态数据成员的例子。#include using namespace std;#include class Student/声明声明Student类类public:Student(char*name=no name);Student();static int getcount()return stu_count;private:static int stu_count;char name40;6.2.2 静态成员函数静态成员函数 Student:Student(char*pname)strcpy(name,pname);st
21、u_count+;/每创建一个对象,学生人数加每创建一个对象,学生人数加1Student:Student()stu_count-;/每销毁一个对象,学生人数减每销毁一个对象,学生人数减1int Student:stu_count=0;6.2.2 静态成员函数静态成员函数 int main()Student s1;couts1.getcount()endl;Student s2;couts2.getcount()endl;coutStudent:getcount()endl;return 0;6.2.2 静态成员函数静态成员函数 6.2.2 静态成员函数静态成员函数 v和静态数据成员一样,和静态
22、数据成员一样,静态成员函数是类的一部分静态成员函数是类的一部分,而不是对象的一部分。,而不是对象的一部分。如果要在类外调用公用的如果要在类外调用公用的静态成员函数,可以用类名和域运算符静态成员函数,可以用类名和域运算符“:”,也允许,也允许通过对象名调用静态成员函数。如:通过对象名调用静态成员函数。如:vStudent:getcount();/用类名调用静态成员函数用类名调用静态成员函数vs1.getcount();/用对象名调用静态成员函数用对象名调用静态成员函数v但这并不意味着此函数是属于对象但这并不意味着此函数是属于对象s1的,而只是用的,而只是用s1的类型而已。的类型而已。6.2.2
23、静态成员函数静态成员函数(1)静态成员函数不能默认访问本类中的非静态成员静态成员函数不能默认访问本类中的非静态成员。当调用。当调用一个对象的成员函数(非静态成员函数)时,系统会把该对一个对象的成员函数(非静态成员函数)时,系统会把该对象的起始地址赋给成员函数的象的起始地址赋给成员函数的this指针。而静态成员函数并不指针。而静态成员函数并不属于某一对象,它与任何对象都无关,因此静态成员函数没属于某一对象,它与任何对象都无关,因此静态成员函数没有有this指针。既然它没有指向某一对象,就无法对一个对象中指针。既然它没有指向某一对象,就无法对一个对象中的非静态成员进行默认访问的非静态成员进行默认访
24、问(即在访问数据成员时不指定对象即在访问数据成员时不指定对象名名)。v 可以说,静态成员函数与非静态成员函数的根本区别是:可以说,静态成员函数与非静态成员函数的根本区别是:非非静态成员函数有静态成员函数有this指针,而静态成员函数没有指针,而静态成员函数没有this指针指针,因因而决定了静态成员函数不能默认访问本类中的非静态成员。而决定了静态成员函数不能默认访问本类中的非静态成员。说明:说明:6.2.2 静态成员函数静态成员函数 v(2)静态成员函数可以直接访问本类中的静态数据成员,静态成员函数可以直接访问本类中的静态数据成员,因因为静态成员同样是属于类的,可以直接访问。在为静态成员同样是属
25、于类的,可以直接访问。在C+程序中,程序中,静态成员函数主要用来访问静态数据成员,而不访问非静态静态成员函数主要用来访问静态数据成员,而不访问非静态成员。假如在一个静态成员函数中有以下语句:成员。假如在一个静态成员函数中有以下语句:v coutageendl;/若若age已声明为已声明为static,则访问本类中的静,则访问本类中的静态成员,合法态成员,合法v coutscoreendl;/若若score是非静态数据成员,不合法是非静态数据成员,不合法说明:说明:6.2.2 静态成员函数静态成员函数 v但是,并不是绝对不能访问本类中的非静态成员,只但是,并不是绝对不能访问本类中的非静态成员,只
26、是不能进行默认访问,因为无法知道应该去找哪个对是不能进行默认访问,因为无法知道应该去找哪个对象。象。如果一定要访问本类的非静态成员,应该加对象如果一定要访问本类的非静态成员,应该加对象名和成员运算符名和成员运算符“.”。如静态成员函数中可以出现:。如静态成员函数中可以出现:vcouts.scoreendl;v这里假设这里假设s已定义为已定义为Student类对象,且在当前作用域类对象,且在当前作用域内有效,则此语句合法。内有效,则此语句合法。说明:说明:#include using namespace std;class Studentpublic:Student(int,int,int);v
27、oid total();static float average();private:int num;int age;float score;static float sum;static int count;Student:Student (int m,int a,int s)num=m;age=a;score=s;void Student:total()sum+=score;count+;float Student:average()return(sum/count);float Student:sum=0;int Student:count=0;静态成员函数的应用静态成员函数的应用int
28、 main()Student stud3=Student(1001,18,70),Student(1002,19,79),Student(1005,20,98);int n;coutn;for(int i=0;in;i+)studi.total();coutThe average score of n;cout students is Student:average()endl;return 0;stud0.average()6.2.2 静态成员函数静态成员函数 v如果想在如果想在average()函数中引用函数中引用stud1中的非静态数据中的非静态数据成员成员score,应该怎样做?,应该
29、怎样做?float Student:average()coutstud1.scoreendl;return(sum/count);float Student:average(Student stu)coutstu.scoreendl;return(sum/count);6.2.2 静态成员函数静态成员函数 v但是在但是在C+程序中最好养成这样的习惯:程序中最好养成这样的习惯:只用静态只用静态成员函数访问静态数据成员,而不访问非静态数据成员函数访问静态数据成员,而不访问非静态数据成员。这样思路清晰,逻辑清楚,不易出错。成员。这样思路清晰,逻辑清楚,不易出错。设计并测试点类设计并测试点类Point,其数据成员是直角坐,其数据成员是直角坐标系的点坐标,友元函数标系的点坐标,友元函数distance用来计算用来计算两点间的距离。两点间的距离。
侵权处理QQ:3464097650--上传资料QQ:3464097650
【声明】本站为“文档C2C交易模式”,即用户上传的文档直接卖给(下载)用户,本站只是网络空间服务平台,本站所有原创文档下载所得归上传人所有,如您发现上传作品侵犯了您的版权,请立刻联系我们并提供证据,我们将在3个工作日内予以改正。