|
《windows環(huán)境多線程編程原理與應(yīng)用》中解釋?zhuān)?如果將類(lèi)的封裝比喻成一堵墻的話,那么友元機(jī)制就像墻上了開(kāi)了一個(gè)門(mén),那些得 到允許的類(lèi)或函數(shù)允許通過(guò)這個(gè)門(mén)訪問(wèn)一般的類(lèi)或者函數(shù)無(wú)法訪問(wèn)的私有屬性和方法。友元機(jī)制使類(lèi)的封裝性得到消弱,所以使用時(shí)一定要慎重。友元類(lèi)的說(shuō)明將外界的某個(gè)類(lèi)在本類(lèi)別的定義中說(shuō)明為友元,那么外界的類(lèi)就成為本類(lèi)的“朋 友”,那個(gè)類(lèi)就可以訪問(wèn)本類(lèi)的私有數(shù)據(jù)了。
class Merchant
{
private :
int m_MyMoney;
int m_MyRoom;
… …
public:
Friend class Lawyer;
Int getmoney();
… …
};
class Lawyer
{
private:
… …
public:
… …
};
只有你賦予某個(gè)類(lèi)為你的友元時(shí),那個(gè)類(lèi)才有訪問(wèn)你的私有數(shù)據(jù)的權(quán)利。
說(shuō)明一個(gè)函數(shù)為一個(gè)類(lèi)的友元函數(shù)則該函數(shù)可以訪問(wèn)此類(lèi)的私有數(shù)據(jù)和方法。定義方法是在類(lèi)的定義中,在函數(shù)名前加上關(guān)鍵字friend.
《挑戰(zhàn)30天C/C++》這樣解釋?zhuān)?br />
在說(shuō)明什么是友元之前,我們先說(shuō)明一下為什么需要友元與友元的缺點(diǎn):
通常對(duì)于普通函數(shù)來(lái)說(shuō),要訪問(wèn)類(lèi)的保護(hù)成員是不可能的,如果想這么做那么必須把類(lèi)的成員都生命成為public(共用的),然而這做帶來(lái)的問(wèn)題遍是任何外部函數(shù)都可以毫無(wú)約束的訪問(wèn)它操作它,c++利用friend修飾符,可以讓一些你設(shè)定的函數(shù)能夠?qū)@些保護(hù)數(shù)據(jù)進(jìn)行操作,避免把類(lèi)成員全部設(shè)置成public,最大限度的保護(hù)數(shù)據(jù)成員的安全。友元能夠使得普通函數(shù)直接訪問(wèn)類(lèi)的保護(hù)數(shù)據(jù),避免了類(lèi)成員函數(shù)的頻繁調(diào)用,可以節(jié)約處理器開(kāi)銷(xiāo),提高程序的效率,但所矛盾的是,即使是最大限度大保護(hù),同樣也破壞了類(lèi)的封裝特性,這即是友元的缺點(diǎn),在現(xiàn)在cpu速度越來(lái)越快的今天我們并不推薦使用它,但它作為c++一個(gè)必要的知識(shí)點(diǎn),一個(gè)完整的組成部分,我們還是需要討論一下的。在類(lèi)里聲明一個(gè)普通數(shù)學(xué),在前面加上friend修飾,那么這個(gè)函數(shù)就成了該類(lèi)的友元,可以訪問(wèn)該類(lèi)的一切成員。
下面我們來(lái)看一段代碼,看看我們是如何利用友元來(lái)訪問(wèn)類(lèi)的一切成員的
//程序作者:管寧
//所有稿件均有版權(quán),如要轉(zhuǎn)載,請(qǐng)務(wù)必著名出處和作者
#include <iostream>
using namespace std;
class Internet
{
public:
Internet(char *name,char *address) // 改為:internet(const char *name , const char *address)
{
strcpy(Internet::name,name);
strcpy(Internet::address,address);
}
friend void ShowN(Internet &obj); //友元函數(shù)的聲明
public: // 改為:private
char name[20];
char address[20];
};
void ShowN(Internet &obj) //函數(shù)定義,不能寫(xiě)成,void Internet::ShowN(Internet &obj)
{
cout<<obj.name<<endl; //可訪問(wèn)internet類(lèi)中的成員
}
void main()
{
Internet a("中國(guó)軟件開(kāi)發(fā)實(shí)驗(yàn)室","www*cndev-lab*com");
ShowN(a);
cin.get();
}
上面的代碼通過(guò)友元函數(shù)的定義,我們成功的訪問(wèn)到了a對(duì)象的保護(hù)成員name,友元函數(shù)并不能看做是類(lèi)的成員函數(shù),它只是個(gè)被聲明為類(lèi)友元的普通函數(shù),所以在類(lèi)外部函數(shù)的定義部分不能夠?qū)懗蓈oid Internet::ShowN(Internet &obj),這一點(diǎn)要注意。
一個(gè)普通函數(shù)可以是多個(gè)類(lèi)的友元函數(shù),對(duì)上面的代碼我們進(jìn)行修改,注意觀察變化:
//程序作者:管寧
//所有稿件均有版權(quán),如要轉(zhuǎn)載,請(qǐng)務(wù)必著名出處和作者
#include <iostream>
using namespace std;
class Country;
class Internet
{
public:
Internet(char *name,char *address) // 改為:internet(const char *name , const char *address)
{
strcpy(Internet::name,name);
strcpy(Internet::address,address);
}
friend void ShowN(Internet &obj,Country &cn);//注意這里
public:
char name[20];
char address[20];
};
class Country
{
public:
Country()
{
strcpy(cname,"中國(guó)");
}
friend void ShowN(Internet &obj,Country &cn);//注意這里
protected:
char cname[30];
};
void ShowN(Internet &obj,Country &cn)
{
cout<<cn.cname<<"|"<<obj.name<<endl;
}
void main()
{
Internet a("中國(guó)軟件開(kāi)發(fā)實(shí)驗(yàn)室","www*cndev-lab*com");
Country b;
ShowN(a,b);
cin.get();
}
一個(gè)類(lèi)的成員函數(shù)函數(shù)也可以是另一個(gè)類(lèi)的友元,從而可以使得一個(gè)類(lèi)的成員函數(shù)可以操作另一個(gè)類(lèi)的數(shù)據(jù)成員,我們?cè)谙旅娴拇a中增加一類(lèi)Country,注意觀察
//程序作者:管寧
//站點(diǎn):www.cndev-lab.com
//所有稿件均有版權(quán),如要轉(zhuǎn)載,請(qǐng)務(wù)必著名出處和作者
#include <iostream>
using namespace std;
class Internet;
class Country
{
public:
Country()
{
strcpy(cname,"中國(guó)");
}
void Editurl(Internet &temp) ;//成員函數(shù)的聲明
protected:
char cname[30];
};
class Internet
{
public:
Internet(char *name,char *address)
{
strcpy(Internet::name,name);
strcpy(Internet::address,address);
}
friend void Country::Editurl(Internet &temp); //友元函數(shù)的聲明
protected:
char name[20];
char address[20];
};
void Country::Editurl(Internet &temp) //成員函數(shù)的外部定義
{
strcpy(temp.address,"edu.cndev-lab*com");
cout<<temp.name<<"|"<<temp.address<<endl;
}
void main()
{
Internet a("中國(guó)軟件開(kāi)發(fā)實(shí)驗(yàn)室","www*cndev-lab*com");
Country b;
b.Editurl(a);
cin.get();
}
整個(gè)類(lèi)也可以是另一個(gè)類(lèi)的友元,該友元也可以稱(chēng)做為友類(lèi)。友類(lèi)的每個(gè)成員函數(shù)都可以訪問(wèn)另一個(gè)類(lèi)的所有成員
//程序作者:管寧
//所有稿件均有版權(quán),如要轉(zhuǎn)載,請(qǐng)務(wù)必著名出處和作者
#include <iostream>
using namespace std;
class Internet;
class Country
{
public:
Country()
{
strcpy(cname,"中國(guó)");
}
friend class Internet; //友類(lèi)的聲明
protected:
char cname[30];
};
class Internet
{
public:
Internet(char *name,char *address)
{
strcpy(Internet::name,name);
strcpy(Internet::address,address);
}
void Editcname(Country &temp);
protected:
char name[20];
char address[20];
};
void Internet::Editcname(Country &temp)
{
strcpy(temp.cname,"中華人民共和國(guó)");
}
void main()
{
Internet a("中國(guó)軟件開(kāi)發(fā)實(shí)驗(yàn)室","www*cndev-lab*com");
Country b;
a.Editcname(b);
cin.get();
}
0
0
0 (請(qǐng)您對(duì)文章做出評(píng)價(jià))
|
|