友元函数与友元类。

 

  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. Unrecognized Windows Sockets error: 0: JVM_Bind 异常怎么办

    Unrecognized Windows Sockets error: 0: JVM_Bind 异常解决办法 java.net.SocketException: Unrecognized Window ...

  2. 《Linux内核设计与实现》读书笔记(十九)- 可移植性

    linux内核的移植性非常好, 目前的内核也支持非常多的体系结构(有20多个). 但是刚开始时, linux也只支持 intel i386 架构, 从 v1.2版开始支持 Digital Alpha, ...

  3. Android 控件架构

    如果说Android上的app是一个有血有肉的人的话,那么人靠衣装马靠鞍,那么控件就是把app装扮的漂漂亮亮的“衣服”.那么安卓的控件到底是如何架构,又是如何渲染的了. 无论是什么控件,在Androi ...

  4. 46 个非常有用的 PHP 代码片段

    在编写代码的时候有个神奇的工具总是好的!下面这里收集了 40+ PHP 代码片段,可以帮助你开发 PHP 项目. 这些 PHP 片段对于 PHP 初学者也非常有帮助,非常容易学习,让我们开始学习吧- ...

  5. ArcMap 连接SDE 出错“Failed to connect to the specified server. Entry for SDE instance no found in services file.”

    问题描述 环境: ARCMAP 10.0 ARCSDE FOR ORACLE 10.0   在通过用ArcMap 连接ORACLE SDE时出现上面的错.   解决方式 在 C:\Windows\Sy ...

  6. C#中 ToString 和 override ToString 的区别

    public class p { public string ToString(){ return "p"; } } public class c:p{ public string ...

  7. 分享我用Qt开发的应用程序【二】在Qt应用程序中使用字体图标fontawesome

    为了使用简单,需要先写一个单件类,头文件的代码如下: 其中静态方法Instance保证IconHelper的实例全局唯一 (注意构造函数已经私有化了) #ifndefICONHELPER_H #def ...

  8. Windows Phone 8 解锁提示IpOverUsbSvc问题——IpOverUsbEnum返回No connected partners found解决方案

    我的1520之前总是无法解锁,提示:IpOverUsbSvc服务没有开启什么的. 根据网上网友的各种解决方案: 1. 把手机时间设置为当前时间,并且关闭“自动设置” 2. 确保手机接入了互联网 3.确 ...

  9. 找出字符串中第一个不重复的字符(JavaScript实现)

    如题~ 此算法仅供参考,小菜基本不懂高深的算法,只能用最朴实的思想去表达. //找出字符串中第一个不重复的字符 // firstUniqueChar("vdctdvc"); --& ...

  10. JAXB命名空间及命名空间前缀处理

    本篇介绍下JAXB进阶使用,命名空间处理 使用package-info.java添加默认命名空间在需要添加命名空间的包下面添加package-info.java文件,然后添加@XmlSchema注解, ...