friend class
C++中以关键字friend声明友元关系。友元可以访问与其有friend关系的类中的私有成员。友元包括友元函数和友元类。
编辑本段1.友元函数
如果在本类以外的其它地方定义了一个函数(这个函数可以是不属于任何类的非成员函数,也可以是其它类的成员函数),在类体中用friend对该函数进行声明,此函数就称为本类的友元函数。一个类的友元函数可以访问这个类中的private成员。
1.1将全局函数声明为友元函数
如果要将一个全局函数(call)声明为本类(Time)的友元函数,则只需要在本类的函数声明部分声明该函数为friend。此时,该函数可以访问本类的private成员。
| class Time{ public: Time(int=1,int=1,int=1); friendvoid call(Time &);//声明友元函数 private: int hour; int min; int sec; }; Time::Time(int h,int m,int s){ hour=h; min=m; sec=s; } void call(Time &t) {//全局函数,且是Time类的友元函数 cout<<"Call:"<<t.hour<<"-"<<t.min<<"-"<<t.sec<<endl;//访问private成员 } int main(){ Time t; call(t); system("PAUSE"); return EXIT_SUCCESS; } |
1.2友元成员函数
如果需要将目标类(Time)中的成员函数(call)声明为本类(Date)的友元函数,则需要在本类的函数声明部分声明该函数为friend。此时,该函数可以访问本类的private成员。
| class Date; //对Date类的提前引用声明 class Time{ public: Time(int=1,int=1,int=1); void call(Date &);//声明成员函数 private: int hour; int min; int sec; }; class Date{ public: Date(int=1,int=1,int=2008); friendvoid Time::call(Date&); //声明Time类的call为本类的友元成员函数 private: int year; int mon; int day; }; Time::Time(int h,int m,int s){ hour=h; min=m; sec=s; } void Time::call(Date &d) { cout<<"TIME:"<<hour<<"-"<<min<<"-"<<sec<<endl; cout<<"DATE:"<<d.mon<<"-"<<d.day<<"-"<<d.year<<endl; //访问Date类的private成员 } Date::Date(int m,int d,int y){ mon=m; day=d; year=y; } int main(){ Time t; Date d; t.call(d); system("PAUSE"); return EXIT_SUCCESS; } |
这里还做了对类的提前引用声明。
1.3关于类的提前引用声明
一般情况下,类必须先声明(给出类体),才能使用。如果需要在类声明之前,使用该类的名字去定义指向该类对象的指针或引用,可以使用提前引用声明。如上例所示,
| class Date; //对Date类的提前引用声明 … void call(Date &);//Date类的引用 … class Date{…}//Date类的声明 |
但不能因为提前引用声明,而去定义一个类的对象,这是不允许的。
| class Date; //紧接着马上定义一个Date类的对象 Date d1;error:aggregate `Date d1' has incomplete type and cannot be defined … class Date{…} |
在定义对象时要为这些对象分配存储空间,在正式声明类之前,编译系统无法确定应为对象分配多大的存储空 间。编译系统只有见到“类体”之后才能确定应该为对象预留多大的空间。所以不能在声明类之前,先定义一个该类的对象。但是可以在声明类之前,先使用该类的 名字定义一个该类的指针或引用。因为指针变量和引用本身的大小是固定的,它与指向的类对象的大小无关。
1.4将一个函数声明为多个类的友元函数
在这种情况下,该函数可以同时访问多个类的private成员。
| class Date; //对Date类的提前引用声明 class Time{ public: Time(int=1,int=1,int=1); friendvoid call(Time&,Date&);//声明函数call为本类的友元成员函数 private: int hour; int min; int sec; }; class Date{ public: Date(int=1,int=1,int=2008); friendvoid call(Time&,Date&); //声明函数call为本类的友元成员函数 private: int year; int mon; int day; }; Time::Time(int h,int m,int s){ hour=h; min=m; sec=s; } Date::Date(int m,int d,int y){ mon=m; day=d; year=y; } void call(Time &t,Date &d) { cout<<"TIME:"<<t.hour<<"-"<<t.min<<"-"<<t.sec<<endl; cout<<"DATE:"<<d.mon<<"-"<<d.day<<"-"<<d.year<<endl; } int main(){ Time t; Date d; call(t,d); system("PAUSE"); return EXIT_SUCCESS; } |
编辑本段2.友元类
可以将一个类(B)声明为当前类(A)的友元。此时,当前类(A)的友元类(B)中的所有成员函数都是当前类的友元函数,可以访问当前类的private成员。
| class Date; //对Date类的提前引用声明 class Time{ public: Time(int=1,int=1,int=1); friendclass Date;//将Date类声明为当前类的友元类 private: int hour; int min; int sec; }; class Date{ public: Date(int=1,int=1,int=2008); void call_hour(Time&); void call_min(Time&); void call_sec(Time&); private: int year; int mon; int day; }; Time::Time(int h,int m,int s){ hour=h; min=m; sec=s; } Date::Date(int m,int d,int y){ mon=m; day=d; year=y; } void Date::call_hour(Time &t){ cout<<"HOUR:"<<t.hour<<endl; } void Date::call_min(Time &t){ cout<<"MINUTE:"<<t.min<<endl; } void Date::call_sec(Time &t){ cout<<"SECOND:"<<t.sec<<endl; } int main(){ Time t; Date d; d.call_hour(t); d.call_min(t); d.call_sec(t); system("PAUSE"); return EXIT_SUCCESS; } |
- 扩展阅读:
-
- 1
注意,
- 2
[1]友元的关系是单向的。如果声明类B是类A的友元类,则类B中的成员函数可以访问类A中的private成员,但类A中的成员函数不能访问类B中的private成员。
- 3
[2]友元的关系不能传递。如果类B是类A的友元类,类C是类B的友元类,不等于类C是类A的友元类。
- 4
Remark:关于谁在前面的问题,就是在声明变量的时候需要对变量进行instantiation,也就是说,对于pointer和reference来说,只需要定义本身的内存地址就好,并没有对类进行解析。而定义class object的时候则需要整个类的实体,也就是defination
- 5
故而在使用类的时候,必须要看到类的实体,也就是说看到类的整个概况。并不需要整个实现。故而,仅仅的class xxx是不行的。也就是使用真个类的declareation。
- 6
不管是friend还是derived。但是对于friend类,可以知道,必须让被friend的类前面有定义。但对于friend function 则必须把引用该类的函数放在后面。
源自:摘抄笔记
- 1
随机推荐
- 单片机上的发光二极管(LED灯)
LED(light-emitting diode),即发光二极管,俗称 LED 小灯,它的种类很多,参数也不尽相同,我们板子上用的是普通的贴片发光二极管.这种二极管通常的正向导通电压是 1.8V到 2 ...
- 写给自己看的Linux运维基础(一) - 系统基础
查看内核版本信息 uname -a 查看发行版本 cat /etc/issue 查看硬件配置 CPU: cat /proc/cpuinfo more /proc/cpuinfo | grep ...
- 亚马逊云服务之CloudFormation
亚马逊的Web Service其实包含了一套云服务.云服务主要分为三种: IaaS: Infrastructure as a service,基础设施即服务. PaaS: Platform as a ...
- fgets函数执行完成后,文件指针如何移动?
用fgets执行之后,读取了文件中的一行,这时文件位置指针是自动指向文件的下一行的开头吗,还是指向当前行的结尾?如果一行的字符串没读取完会怎样? 实例结果: 如果一行的字符串没读取完会,下一次会接着上 ...
- 基于RequireJS和JQuery的模块化编程——常见问题解析
由于js的代码逻辑越来越重,一个js文件可能会有上千行,十分不利于开发与维护.最近正在把逻辑很重的js拆分成模块,在一顿纠结是使用requirejs还是seajs的时候,最终还是偏向于requirej ...
- ubuntu上安装mysql 编译安装
为什么要折腾?首先说明的是ubuntu上安装mysql等软件是非常容易简单的,其简单的程度盖过windows上的安装,一句sudo apt-get install就可以搞定.如果想用最简便的方法安装m ...
- 更新日志 - fir.im 主题壁纸来了
fir.im 产品开发团队最近主要在优化应用管理后台和 BugHD 后台,新版应用管理后台很快会与大家见面. 本周其他更新内容简单概述如下: 1.fir.im 工具页添加壁纸主题包 有很多用户很喜欢 ...
- javaweb学习总结(六)——Servlet开发(二)
一.ServletConfig讲解 1.1.配置Servlet初始化参数 在Servlet的配置文件web.xml中,可以使用一个或多个<init-param>标签为servlet配置一些 ...
- 生成读取相关连接的物理地址的lib(动态导入库)和dll(动态链接库)
一.导出相关dll库 将原先的CmdInfoToPipe.h class后加入关键字 __declspec(dllexport) #ifndef NETINFO_CMDINFOTOPIPE_H_ #d ...
- Vue.js:轻量高效的前端组件化方案(转载)
摘要:Vue.js通过简洁的API提供高效的数据绑定和灵活的组件系统.在前端纷繁复杂的生态中,Vue.js有幸受到一定程度的关注,目前在GitHub上已经有5000+的star.本文将从各方面对Vue ...