1、为什么要引入友元函数:在实现类之间数据共享时,减少系统开销,提高效率

      具体来说:为了使其他类的成员函数直接访问该类的私有变量

      即:允许外面的类或函数去访问类的私有变量和保护变量,从而使两个类共享同一函数

      优点:能够提高效率,表达简单、清晰

      缺点:友元函数破环了封装机制,尽量不使用成员函数,除非不得已的情况下才使用友元函数。

2、什么时候使用友元函数:

      1)运算符重载的某些场合需要使用友元。

      2)两个类要共享数据的时候

3、怎么使用友元函数:

友元函数的参数:

       因为友元函数没有this指针,则参数要有三种情况:

       1、  要访问非static成员时,需要对象做参数;--常用(友元函数常含有参数)

       2、  要访问static成员或全局变量时,则不需要对象做参数

       3、  如果做参数的对象是全局对象,则不需要对象做参数

友元函数的位置:

       因为友元函数是类外的函数,所以它的声明可以放在类的私有段或公有段且没有区别。

友元函数的调用:

       可以直接调用友元函数,不需要通过对象或指针

友元函数的分类:

根据这个函数的来源不同,可以分为三种方法:

1、普通函数友元函数:

       a) 目的:使普通函数能够访问类的友元

       b) 语法:声明位置:公有私有均可,常写为公有

                        声明: friend + 普通函数声明

                        实现位置:可以在类外或类中

                        实现代码:与普通函数相同(不加不用friend和类::)

                        调用:类似普通函数,直接调用

       c) 代码:        

  1. class INTEGER  
  2. {    
  3. private:  
  4.     int num;  
  5. public:  
  6.     friend void Print(const INTEGER& obj);//声明友元函数  
  7. };  
  8. void Print(const INTEGER& obj)//不使用friend和类::  
  9. {  
  10.     //函数体  
  11. }  
  12. void main()  
  13. {  
  14.     INTEGER obj;  
  15.     Print(obj);//直接调用  
  16. }  

2、类Y的所有成员函数都为类X友元函数—友元类

      a)目的:使用单个声明使Y类的所有函数成为类X的友元

                        它提供一种类之间合作的一种方式,使类Y的对象可以具有类X和类Y的功能

                        具体来说:

                                前提:A是B的友元(=》A中成员函数可以访问B中有所有成员,包括私有成员和公有成员--老忘)

                                    则:在A中,借助类B,可以直接使用~B . 私有变量~的形式访问私有变量

      b)语法:声明位置:公有私有均可,常写为私有(把类看成一个变量)

                        声明: friend + 类名---不是对象啊

                        调用:

      c)代码:

  1. class girl;  
  2.   
  3. class boy  
  4. {    
  5. private:  
  6.     char *name;    
  7.     int age;    
  8. public:    
  9.     boy();  
  10.     void disp(girl &);     
  11. };    
  12.   
  13. void boy::disp(girl &x) //函数disp()为类boy的成员函数,也是类girl的友元函数   
  14. {   
  15.     cout<<"boy's name is:"<<name<<",age:"<<age<<endl;//正常情况,boy的成员函数disp中直接访问boy的私有变量  
  16.     cout<<"girl's name is:"<<x.name<<",age:"<<x.age<<endl;   
  17.     //借助友元,在boy的成员函数disp中,借助girl的对象,直接访问girl的私有变量  
  18.     //正常情况下,只允许在girl的成员函数中访问girl的私有变量  
  19. }  
  20.   
  21. class girl  
  22. {    
  23. private:  
  24.     char *name;    
  25.     int age;    
  26.     friend boy;   //声明类boy是类girl的友元    
  27. public:    
  28.     girl();     
  29. };    
  30. void main()    
  31. {     
  32.     boy b;    
  33.     girl g;    
  34.     b.disp(g);  //b调用自己的成员函数,但是以g为参数,友元机制体现在函数disp中  
  35. }  

3、类Y的一个成员函数为类X的友元函数

      a)目的:使类Y的一个成员函数成为类X的友元

             具体而言:而在类Y的这个成员函数中,借助参数X,可以直接以X。私有变量的形式访问私有变量

      b)语法:声明位置:声明在公有中 (本身为函数)

                        声明:friend + 成员函数的声明

                        调用:先定义Y的对象y---使用y调用自己的成员函数---自己的成员函数中使用了友元机制

      c)代码: 

  1. class girl;   
  2. class boy  
  3. {    
  4. private:  
  5.     char *name;    
  6.     int age;    
  7. public:    
  8.     boy();  
  9.     void disp(girl &);       
  10. };     
  11.    
  12. class girl  
  13. {  
  14. private:  
  15.     char *name;    
  16.     int age;    
  17. public:    
  18.     girl(char *N,int A);    
  19.     friend void boy::disp(girl &); //声明类boy的成员函数disp()为类girl的友元函数    
  20. };    
  21.    
  22. void boy::disp(girl &x)    
  23. {     
  24.     cout<<"boy's name is:"<<name<<",age:"<<age<<endl;  //访问自己(boy)的对象成员,直接访问自己的私有变量    
  25.     cout<<"girl's name is:"<<x.name<<",age:"<<x.age<<endl;    
  26.     //借助友元,在boy的成员函数disp中,借助girl的对象,直接访问girl的私有变量  
  27.     //正常情况下,只允许在girl的成员函数中访问girl的私有变量    
  28. }    
  29. void main()    
  30. {     
  31.     boy b();    
  32.     girl g();    
  33.     b.disp(g);  }  

4、在模板类中使用友元operator<<(对<<运算符的重载)

    a)使用方法:

在模板类中声明:

  1. friend ostream& operator<< <>(ostream& cout,const MGraph<VexType,ArcType>& G);  

在模板类中定义:

  1. template<class VexType,class ArcType>  
  2. ostream& operator<<(ostream& cout,const MGraph<VexType,ArcType>& G)  
  3. {  
  4.     //函数定义  
  5. }  

    b)注意:

把函数声明非模板函数:

  1. friend ostream& operator<< (ostream& cout,const MGraph& G);  

把函数声明为模板函数:

  1. friend ostream& operator<< <>(ostream& cout,const MGraph<VexType,ArcType>& G);  

或:

  1. friend ostream& operator<< <VexType,ArcType>(ostream& cout,const MGraph<VexType,ArcType>& G);  

说明:
 在函数声明中加入operator<< <>:是将operator<<函数定义为函数模板,将函数模板申明为类模板的友员时,是一对一绑定的
 实际的声明函数:这里模板参数可以省略,但是尖括号不可以省略

  1. friend ostream& operator<< <VexType,ArcType>(ostream& cout,const MGraph<VexType,ArcType>& G);  

5、友元函数和类的成员函数的区别:成员函数有this指针,而友元函数没有this指针。

6、记忆:A是B的友元《=》A是B的朋友《=》借助B的对象,在A中可以直接 通过B。成员变量(可以是公有,也可以为私有变量) 的方式访问B 

C++_友元函数的更多相关文章

  1. C++_友元函数总结(转)

     原文地址:http://blog.csdn.net/insistgogo/article/details/6608672   1.为什么要引入友元函数:在实现类之间数据共享时,减少系统开销,提高效率 ...

  2. C++_友元函数(转)

    1.为什么要引入友元函数:在实现类之间数据共享时,减少系统开销,提高效率 具体来说:为了使其他类的成员函数直接访问该类的私有变量 即:允许外面的类或函数去访问类的私有变量和保护变量,从而使两个类共享同 ...

  3. 009_linuxC++之_友元函数

    (一)定义:友元函数是指某些虽然不是类成员却能够访问类的所有成员的函数.类授予它的友元特别的访问权.通常同一个开发者会出于技术和非技术的原因,控制类的友元和成员函数(否则当你想更新你的类时,还要征得其 ...

  4. String类型_static成员_动态内存分配_拷贝构造函数_const关键字_友元函数与友元类

    1:String类型 #include <iostream> using namespace std; int main() { //初始化方法 string s1 = "hel ...

  5. C++的友元类和友元函数实例

    #include <math.h> #include<iostream> using namespace std; class Point { public: Point(do ...

  6. c++友元函数

    c++友元函数分两类: 一://友员全居函数 /*#include <iostream>using namespace std;class aaa{    friend void prin ...

  7. 重载运算符:类成员函数or友元函数

    类成员函数: bool operator ==(const point &a)const { return x==a.x; } 友元函数: friend bool operator ==(co ...

  8. 不可或缺 Windows Native (20) - C++: 友元函数, 友元类

    [源码下载] 不可或缺 Windows Native (20) - C++: 友元函数, 友元类 作者:webabcd 介绍不可或缺 Windows Native 之 C++ 友元函数 友元类 示例演 ...

  9. [Reprint]C++友元函数与拷贝构造函数详解

    这篇文章主要介绍了C++友元函数与拷贝构造函数,需要的朋友可以参考下   一.友元函数 1.友元函数概述: (1)友元函数是定义在一个类外的普通函数.友元函数和普通函数的定义一样;在类内必须将该普通函 ...

随机推荐

  1. [bzoj4815]: [Cqoi2017]小Q的表格

    来自FallDream的博客,未经允许,请勿转载,谢谢. 小Q是个程序员. 作为一个年轻的程序员,小Q总是被老C欺负,老C经常把一些麻烦的任务交给小Q来处理.每当小Q不知道如何解决时,就只好向你求助. ...

  2. bzoj3223Tyvj 1729 文艺平衡树 splay

    3223: Tyvj 1729 文艺平衡树 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 5644  Solved: 3362[Submit][Sta ...

  3. 【集训第二天·翻水的老师】--ac自动机+splay树

    今天是第二天集训.(其实已经是第三天了,只是昨天并没有机会来写总结,现在补上) 上午大家心情都很愉快,因为老师讲了splay树和ac自动机. 但到了下午,我们的教练竟然跑出去耍了(excuse me? ...

  4. python安装pip以及导入第三方包

    python有着强大的第三方库,数量很多且功能强大. 最原始的办法是在官网上下载压缩包,解压,然后运行setup.py来进行安装. 显然这种方法很繁琐,不方便.因此有了包管理工具. pip是一个包管理 ...

  5. jquery传值与判断

    js判断是否包含字符串 var str="Hello world!" var s = str.indexOf("Hello") 存在则s>-1不存在则是s ...

  6. WebService之soap类型的服务和rest类型的服务

    1.引言 WebService顾名思义就是web服务,web服务主要有两种,一种是基于soap类型的服务,一种是基于rest类型的服务,其中soap类型的服务有两种版本,一种是soap1.1版本,一种 ...

  7. Lucene——Field.Store(存储域选项)及Field.Index(索引选项)

    Field.Store.YES或者NO(存储域选项) 设置为YES表示或把这个域中的内容完全存储到文件中,方便进行文本的还原 设置为NO表示把这个域的内容不存储到文件中,但是可以被索引,此时内容无法完 ...

  8. fireBug引入JQuery,方便书写jq调试代码

    在控制台执行下段代码,等到网络中加载完成后,即可正常运行jq代码.也可以根据需要进行修改引入其他js代码. javascript:(function(url) { var s = document.c ...

  9. 在web应用中使用Log4j 2

    Using Log4j 2 inWeb Applications (在web应用中使用Log4j 2) 来源:http://logging.apache.org/log4j/2.x/manual/we ...

  10. Java对象的访问定位

    java对象在访问的时候,我们需要通过java虚拟机栈的reference类型的数据去操作具体的对象.由于reference类型在java虚拟机规范中只规定了一个对象的引用,并没有定义这个这个引用应该 ...