C++中友元函數(shù)(friend)解析
文章轉(zhuǎn)自公眾號:Coder梁(ID:Coder_LT)
我們知道C++控制對象的私有部分的訪問,只能通過公共的接口。這樣的設(shè)計當(dāng)然沒錯,但有的時候也會顯得過于嚴(yán)格,產(chǎn)生一些問題。
因此C++提供了另外一種形式的訪問權(quán)限,叫做友元(friend
)。
友元有三種,分別是友元函數(shù)、友元類和友元成員函數(shù)。
通過讓函數(shù)成為類的友元,可以賦予該函數(shù)與類成員函數(shù)一樣的訪問權(quán)限,也就是說我們可以在友元函數(shù)當(dāng)中訪問類的私有成員變量。
在介紹友元函數(shù)的使用之前,我們需要先了解為什么需要友元函數(shù)。C++ Primer
中給了一個非常不錯的例子,在之前運算符重載的例子當(dāng)中,我們實現(xiàn)了一個類Time
。用來記錄時間,假設(shè)我們需要重載它的*運算符,能夠允許一個時間對象和一個浮點數(shù)相乘。
很明顯,我們只需要重載運算符*即可:
Time Time::operator*(const double x) { ? ? // todo }
我們在使用的時候大概是這樣:
Time a, b; a = b * 32.5;
但是這里有一個小問題,我們寫成a = b * 32.5
;可以,但如果反過來寫成32.5 * b
就不行了。因為對于b * 32.5
來說本質(zhì)上是b調(diào)用了operator
*函數(shù),等價于a = b.opeartor*(32.5)
;。但后者就不行了,要怎么解決呢,只能另外實現(xiàn)一個函數(shù)來解決了,這個函數(shù)有兩個input,分別是double
和Time
類型,返回一個Time
類型。
Time operator*(double m, const Time &t);
但這又有了新的問題,由于這不是一個成員函數(shù),不能直接訪問類的私有數(shù)據(jù)。為了破例讓它能夠訪問,我們需要將它設(shè)置成友元。
創(chuàng)建友元的方法很簡單,我們只需要在函數(shù)簽名之前加上關(guān)鍵字friend
。
friend Time operator*(double m, const Time &t);
它有兩個含義:
- 它不是成員函數(shù),因此不能使用成員函數(shù)運算符來調(diào)用
- 它與成員函數(shù)的訪問權(quán)限相同,即可以訪問所有
private
和public
數(shù)據(jù)
由于友元函數(shù)不是成員函數(shù),所有我們在實現(xiàn)的時候不需要使用Time::
限定符,也不用在實現(xiàn)當(dāng)中加上關(guān)鍵字friend
,
函數(shù)的實現(xiàn)如下:
Time operator*(double m, const Time &t) { ? ? Time result; ? ? long totalminutes = t.hours * m * 60 + t.minutes * m; ? ? result.hours = totalminutes / 60; ? ? result.minutes = totalminutes % 60; ? ? return result; }
我們在函數(shù)當(dāng)中直接訪問了hours
和minutes
成員變量,因此函數(shù)必須是友元函數(shù)。
當(dāng)然我們可以把函數(shù)稍微變換一下,就可以不必是友元函數(shù)了:
Time operator*(double m, const Time &t) { ? ? return t * m; // ?調(diào)用了t.operator*(m) }
在這個函數(shù)當(dāng)中,我們沒有顯式地訪問私有變量,因此可以不必是友元。
到此這篇關(guān)于C++友元函數(shù)講解的文章就介紹到這了,更多相關(guān)C++友元函數(shù)內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C/C++?Qt數(shù)據(jù)庫與SqlTableModel組件應(yīng)用教程
SqlTableModel?組件可以將數(shù)據(jù)庫中的特定字段動態(tài)顯示在TableView表格組件中,這篇文章將主要介紹SqlTableModel組件一些常用的操作,需要的朋友可以參考一下2021-12-12