天天看點

友元函數&友元類

    C++類具有封裝和資訊隐蔽的特性,隻有類中的成員函數才能通路類的私有成員,類外無法通路。

    存在問題:當需要多次存取類的資料成員時,會出現相應的成員函數被多次調用的情況。由于參數傳遞、類類型檢查和安全性檢查等都需要時間開銷,進而将影響程式的運作效率。但如果将這些資料成員都定義為公有,破壞了隐蔽的特性。

    解決辦法:友元函數。

    友元函數在類中聲明,但是不屬于該類的成員函數。如果要在它的函數體中操縱類的資料成員,必須通過參數傳遞該類的對象來進行。

//友元函數
#include<iostream>
#include<cmath>
using namespace std;

class CPoint
{
public:
	CPoint(double xx=0,double yy=0);
	double getX();
	double getY();
	friend double fDist(CPoint *p1,CPoint *p2);
private:
	double x,y;
};

CPoint::CPoint(double xx,double yy)
{
	x=xx;
	y=yy;
}

double CPoint::getX()
{
	return x;
}

double CPoint::getY()
{
	return y;
}

double fDist(CPoint *p1,CPoint *p2)//指針
{
	double x=double(p1->x-p2->x);
	double y=double(p1->y-p2->y);

	return sqrt(x*x+y*y);
}

double fDist_s(CPoint *p1,CPoint *p2)//非友元,不能通過CPoin類對象指針直接通路所指對象的私有資料成員x與y
{
	double x=double(p1->getX()-p2->getX());//因為是非友元,是以隻能通過getX(),getY()來通路私有資料成員了
	double y=double(p1->getY()-p2->getY());

	return sqrt(x*x+y*y);
}

int main()
{
	CPoint *p1=new CPoint(1,2);
	CPoint *p2=new CPoint(2,4);

	cout<<"通過友元函數:";
	cout<<"the distance is:"<<fDist(p1,p2)<<endl;

	cout<<"通過普通函數:";
	cout<<"the distance is:"<<fDist_s(p1,p2)<<endl;

	delete p1;
	delete p2;

	system("pause");
	return 0;
}
           

由于定義了友元函數fDist(),可以通過CPoint類對象指針p1,p2直接通路所指對象的私有資料成員x和y。

由于fDist_s()非友元,不能通過CPoin類對象指針直接通路所指對象的私有資料成員x與y,是以隻能通過getX(),getY()來通路私有資料成員了。

友元函數&amp;友元類
//友元類
#include<iostream>
#include<string>
using namespace std;

class B;//先聲明類B
class A
{
private:
	friend class B;
	char *name;
	int age;
public:
	A(char *str,int n);
};
class B
{
public:
	void show(A *x);
};

A::A(char *str,int n)
{
	name=new char[strlen(str)+1];
	strcpy(name,str);
	age=n;
}

//void B::show(A x)
void B::show(A *x)//由于類A把類B聲明為它的友元類,是以類B的成員函數相應的都成為了類A的友元函數
{
	cout<<"姓名:"<<x->name<<endl;
	cout<<"年齡:"<<x->age<<endl;
}

int main()
{
	//A a("蘭智輝",21);
	A *a=new A("蘭智輝",21);
	B b;
	b.show(a);

	system("pause");
	return 0;
}
           

由于類A把類B聲明為它的友元類,是以類B的成員函數相應的都成為了類A的友元函數,可以自由的通路類A的私有資料成員name和age。

繼續閱讀