• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            隨筆-145  評論-173  文章-70  trackbacks-0

            繼承是一個永恒的話題,可是帶來的問題卻也是多多啊。最近一直在思考一些細節的東西,于是也有了不少收獲。今天,當我想要用派生類訪問基類的私有成員時,發現問題了……
            代碼1:(沒有問題,對照下面的代碼來看錯誤之處)

             1#include <iostream>
             2using namespace std;
             3class B;
             4class A{
             5int b;
             6public:
             7  friend class B;
             8  int a;
             9  A(int x):a(x) { b = a+1;}
            10}
            ;
            11class B:public A{
            12public:
            13  void print();
            14    B(int x):A(x){} //基類定義了一個有參的構造函數時,派生類必須定義構造函數 
            15}
            ;
            16void B::print()
            17
            18  cout << B::a << endl;
            19  cout << B::b << endl;
            20}

            21
            22int main()
            23{
            24B b(4);
            25b.print();
            26}

            27
            這里可以看到,結果是正確的,而且還能夠訪問到基類的私有成員b,證明了自己的想法。其中,將派生類都作為了基類的友元,這個范圍未免有點大了,好吧,我縮小點,(問題來了……)
            問題代碼:
             1#include <iostream>
             2using namespace std;
             3class B;
             4class A{
             5int b;
             6public:
             7  int a;
             8  A(int x):a(x) { b = a+1;}
             9  friend void B::print(); 
            10}
            ;
            11class B:public A{
            12public:
            13  void print();
            14  B(int x):A(x){} //基類定義了一個有參的構造函數時,派生類必須定義構造函數 
            15}
            ;
            16
            17
            18void B::print()
            19
            20  cout << B::a << endl;
            21  cout << B::b << endl;
            22}

            23
            24int main()
            25{
            26B b(4);
            27b.print();
            28}
            變化后的代碼僅僅就是將類B的print函數作為基類A的友元,這樣,按照道理來說,應該也是可以訪問基類的私有成員的。而且從上面那個正確的代碼來看,聲明為友元之后,確實是可以突破訪問權限的,然后,此代碼有錯誤,顯示的就是私有成員的不可訪問。
            一時間感覺世界有點不真實了…………

            -------------------------------目前原因正在思考中,如果哪個過客知道答案,后者有自己的見解,不妨占用下你寶貴的時間,留個言吧。

            (注記:個人認為,印象中《C++ Primer》中講到在聲明類的成員函數為另外一個類的友元時,需要先定義這個類,而如果是派生類的話,可以聲明下,等后來再定義。這個也是代碼1的做法,沒有問題。所以我覺得可能是如果用下面的代碼,那么沒有先定義類B,而聲明了它的成員函數print,所以有問題。待求證……)




            結束:
            感謝Sunshine Alike  ,我覺得他的解釋可能是比較合理的,摘抄如下,如果有誤,歡迎留言指正。。
            代碼2中,僅僅聲明類B是無法得知類內部具體實現的,所以你這個時候指定
            B中的一個函數作為友元是不行的,因為在這個時候編譯器還不知道B的具體實現是怎么樣的。
            代碼1:開始時聲明類B,將其整個類作為友元類,編譯器無需知道其內部實現,所以可以通過

            posted on 2010-01-12 23:53 deercoder 閱讀(4189) 評論(10)  編輯 收藏 引用

            評論:
            # re: C++友元的一個問題-----------由派生類訪問基類的私有成員 2010-01-13 01:15 | zhaoyg
            剛才在看到friend void B::print(); 這句時就已經覺得問題就是由他引發的。
            因為我記得如果是要將類的成員作為友元,那該類就應該是已定義的。

            經編譯,得到驗證
            報錯如下:
            error: member `void B::print()' declared as friend before type `B' defined
            C++ Primer 上也說得是“必須先定義一個類,才能將成員函數設為友元”

            還有,
            “而如果是派生類的話,可以聲明下,等后來再定義”
            這句話我沒有在C++ Primer 上找到。  回復  更多評論
              
            # re: C++友元的一個問題-----------由派生類訪問基類的私有成員 2010-01-13 10:39 | 劉暢
            @zhaoyg
            謝謝你的回答。
            對于這個錯誤,我開始的看法和你是一樣的,認為必須先要定義一個類,然后再可以聲明它的函數是另一個類的友元。
            可是,看看前面的代碼1我就覺得有點不對,因為對于class B,先開始的時候只是聲明,而沒有定義,這樣卻可以將整個類作為它的友元,我覺得也有點不可行。不然,這樣不是和上面的一個道理嗎?  回復  更多評論
              
            # re: C++友元的一個問題-----------由派生類訪問基類的私有成員 2010-01-13 11:18 | Sunshine Alike
            @劉暢
            代碼2中,僅僅聲明類B是無法得知類內部具體實現的,所以你這個時候指定
            B中的一個函數作為友元是不行的,因為在這個時候編譯器還不知道B的具體實現是怎么樣的。
            代碼1:開始時聲明類B,將其整個類作為友元類,編譯器無需知道其內部實現,所以可以通過  回復  更多評論
              
            # re: C++友元的一個問題-----------由派生類訪問基類的私有成員 2010-01-13 12:31 | 劉暢
            @Sunshine Alike
            有道理,其實前面也覺得可能跟類的實現有關吧,學習了~~~  回復  更多評論
              
            # re: C++友元的一個問題-----------由派生類訪問基類的私有成員 2010-09-21 07:39 | @kll
            友元的訪問權限 與類的繼承你搞混了  回復  更多評論
              
            # re: C++友元的一個問題-----------由派生類訪問基類的私有成員 2010-09-21 07:51 | @kll
            原來不注冊可以留言 呵呵 上面說的只是測試 下面說下我理解不知道對不對
            code1 中將類B設為A的友元 B又是A的public繼承 那么B能訪問A的private成員是因為友元的緣故 B::b等價于 B:: A::b 故B.print() 可以執行
            code2 只是將B的成員print()設為A的友元 那么B::b 則不成立 無法訪問吧
            若是 print 函數帶A對象的引用會通過的

            我了解的這一點不多 其實我是想查詢關于公有派生類的友元 能不能訪問基類的private的成員 或者訪問權限到那一層 在繼承層次中友元的權限又有哪些 這方面的文章 如果斑竹有資料 望不吝賜教 謝謝 忘了打標點 估計看起來麻煩 呵呵 我沒有代碼運行的環境 要不然能測試了 呵呵  回復  更多評論
              
            # re: C++友元的一個問題-----------由派生類訪問基類的私有成員 2014-01-12 12:51 | panpan
            為什么我將第一種的 A(int x):a(x) { b = a+1;}改為A(int x):a(x),b(a+1) {}運行結果為
            4
            -858993459
            請按任意鍵繼續. . .

            為什么 搞不清楚  回復  更多評論
              
            # re: C++友元的一個問題-----------由派生類訪問基類的私有成員 2014-01-12 12:53 | panpan
            哦哦 搞懂了 成員變量是按照在類中聲明的先后順序來初始化的   回復  更多評論
              
            # re: C++友元的一個問題-----------由派生類訪問基類的私有成員 2014-01-12 13:32 | panpan
            我覺得這個派生類成員函數作為基類的成員友元本身就有問題,首先,類B是類A的派生類,那么類A必須在類A前定義,而派生類B的函數print()是基類友元,使得派生類B必須先于A定義(不是聲明),這兩個前提條件已經互相矛盾,所以怎么都是有問題的  回復  更多評論
              
            # re: C++友元的一個問題-----------由派生類訪問基類的私有成員 2014-01-12 13:56 | panpan
            我試了 第二種情況 即便是傳給print()一個A&參數,照樣是不能訪問b成員變量的 這個確實沒想通  回復  更多評論
              
            久久久人妻精品无码一区| 一本一本久久A久久综合精品| 丰满少妇高潮惨叫久久久| 久久青青草原综合伊人| 久久精品中文字幕第23页| 久久综合视频网| 天天久久狠狠色综合| 久久精品国产精品亚洲精品| 久久ww精品w免费人成| 色综合合久久天天给综看| 久久精品一本到99热免费| 久久久噜噜噜久久中文字幕色伊伊| 狠狠综合久久AV一区二区三区| 精品久久久久久国产91| 久久精品桃花综合| 亚洲国产精品热久久| 久久人人妻人人爽人人爽| 一本色道久久综合狠狠躁篇| 女人香蕉久久**毛片精品| 久久久久av无码免费网| 久久久久18| 国内精品久久久久久麻豆| 九九精品99久久久香蕉| 亚洲中文字幕无码久久综合网 | 亚洲国产另类久久久精品小说| 99久久国产亚洲高清观看2024 | 国产精品99久久免费观看| 中文字幕亚洲综合久久菠萝蜜| 99久久精品国产一区二区蜜芽| 一本一道久久综合狠狠老| 久久久久久国产精品无码下载| 欧美久久天天综合香蕉伊| 国产精品欧美久久久久天天影视 | 久久久久亚洲av无码专区| 久久精品国产99国产精品亚洲| 亚洲午夜精品久久久久久app| 国产AⅤ精品一区二区三区久久| 久久综合久久综合久久| 久久久九九有精品国产| 超级碰久久免费公开视频| 国产精品美女久久久久AV福利 |