友元函数的产生和理解

        最近在网上看了一些友元的资料,发现你们写的都很好,想转载一下文章,但是有几点不太好(一、叙述的太多了,看完容易忘了【沙米比较容易健忘】 二、只是知识的讲解,以为比较枯燥【哈哈,本身的感觉】)。下面是本身作的一些改进。ios

        进入正题啦。下面按照为什么须要友元函数友元类怎么使用使用的地方?来讲说友元函数的本质。编程

    为何须要友元函数和友元类   框架

     你们学过JAVA的话,应该知道其反射的机制吧,经过反射能够逆向访问类的私有成员和函数。从而衍生出了Web框架的AOP(面向切面编程)等,极大的方便了开发。函数

     JAVA有,并且好用。C++固然也要有的,因而友元函数和友元类就产生了,本质就是类外的其余函数或其余类也能够访问该类的私有变量。从而实现类之间数据共享。spa

凡事有利就有弊,在使用其时,没有必要使用,尽可能不要使用,不要过多的破坏面向对象的封装性。code

       优势:可以提升效率,表达简单、清晰对象

        缺点:友元函数破环了封装机制,是程序的逻辑变得复杂和混乱。继承

 

    怎么使用友元函数和友元类ip

     友元函数使用:ci

‍     普通友元函数: 在类A的里面声明此友元函数,在类A的外面定义此友元函数(访问某类的私有成员 )

     类B的成员函数做为类A的友元函数:先定义类B,而后在类A中声明类B的成员函数为其友元函数,最后定义类B的成员函数

    友元类  :类B的是类A的友元类,在类A中声明,在类B能够经过成员(实例化A的对象),访问A的私有成员

     注意:一、友元函数的声明语句位置与访问描述(public、private)无关。

                二、友元函数经过对象参数访问私有数据成员

   流程以下:

    代码以下:

    普通友元函数:

#include <iostream>
using namespace std;

class friendClass
{
public:
	friendClass(int m, int n)
	{
		m_a = m;
		m_b = n;
	}
friend void printClass(friendClass& fcls);    //友元函数声明处

private:
	int m_a;
	int m_b;
};

void printClass(friendClass& fcls)    //友元函数定义处
{
	cout<<"a="<<fcls.m_a<<"  b="<<fcls.m_b<<endl;
}

int main()
{
    friendClass ff(1,2);
    printClass(ff);
}

  类的成员函数作友元函数

    注意顺序成员函数做为友元函数的类定义要在友元函数访问的类定义以前。

    缘由是,只有当一个类的定义已经被看到时它的成员函数才能被声明为另外一个类的友元(很重要

    tip:声明能够有多处,定义只能有一处,在编译器已经加载了定义,其余的声明才起做用

#include <iostream>
using namespace std;

class friendClass; 

class friendFuncClass    //成员函数做为友元函数的类
{
public:
	void printClass(friendClass& fcls);    //声明友元函数
};

class friendClass       //友元函数访问的类
{
public:
	friendClass(int m, int n)
	{
		m_a = m;
		m_b = n;
	}
friend	void friendFuncClass::printClass(friendClass& fcls);//声明友元函数

private:
	int m_a;
	int m_b;
};

void friendFuncClass::printClass(friendClass& fcls)    //定义友元函数
{
	cout<<"a="<<fcls.m_a<<"  b="<<fcls.m_b<<endl;
}

int main()
{
    friendClass ff(1,2);
    friendFuncClass fu;
    fu.printClass(ff);
    
    system("pause");
    return 0;
}

    友元类   注意:

      (1) 友元关系不能被继承。 

      (2) 友元关系是单向的,不具备交换性。若类B是类A的友元,类A不必定是类B的友元,要看在类中是否有相应的声明。 

      (3) 友元关系不具备传递性。若类B是类A的友元,类C是B的友元,类C不必定是类A的友元,一样要看类中是否有相应的声明

#include <iostream>
using namespace std;

//友元类的使用,类B的是类A的友元类,类B能够经过成员(实例化A的对象),访问A的私有成员
class frindAClass
{
public:
	frindAClass()
	{
		A_a = 1;
		A_b = 2;
	}
private:
	int A_a;
	int A_b;
	friend class frindBClass;
};


class frindBClass
{
public:
	frindBClass(frindAClass& fa ):objA(fa)
	{

	}

	void printA()    //打印类A的私有成员
	{
		cout<<"a="<<objA.A_a<<"  b="<<objA.A_b<<endl;
	}
private:
	frindAClass objA;
};

int main()
{
	//友元类的使用
	frindAClass fa;
	frindBClass fb(fa);
	fb.printA();    

	system("pause");
	return 0;
}

   

使用的地方     

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

           如:须要重载输入输出运算符(<< 、>>),这个双目运算符的左侧运算量是 cin或 cout,不是对象自己,咱们就只能使用友元函数来访问对象的私有成员。

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

最后但愿能对你们有帮助,沙米才疏学浅,有什么错误请留言指正,谢谢你们。

相关文章
相关标签/搜索