• <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成員變量的 這個確實沒想通  回復  更多評論
              
            99久久精品国产一区二区三区| 国产精品成人99久久久久 | 国产综合久久久久| 91精品国产乱码久久久久久| 成人a毛片久久免费播放| 婷婷久久综合九色综合绿巨人| 狠狠精品久久久无码中文字幕| 久久99精品久久久久久| 久久夜色精品国产www| 久久夜色精品国产噜噜亚洲AV| 久久婷婷综合中文字幕| 亚洲精品综合久久| 久久精品国产影库免费看| 香蕉久久夜色精品国产尤物| 国产精品久久久久国产A级| 日韩欧美亚洲综合久久影院Ds| 性欧美大战久久久久久久久| 久久精品成人免费观看97| 久久亚洲私人国产精品vA| 久久亚洲欧洲国产综合| 日本三级久久网| 久久久久久亚洲AV无码专区| 久久这里有精品视频| 国产精品伦理久久久久久| 国产精品美女久久久m| 伊人久久精品无码二区麻豆| 青草久久久国产线免观| 亚洲国产二区三区久久| 97热久久免费频精品99| 日本久久久久亚洲中字幕| 日本精品一区二区久久久| 99久久精品免费看国产一区二区三区| 亚洲国产精品无码久久久秋霞2 | 久久久国产打桩机| 综合久久一区二区三区| 亚洲人成无码网站久久99热国产 | 久久夜色精品国产噜噜亚洲a| 久久久久亚洲AV综合波多野结衣| 日本精品久久久久中文字幕8| 99久久精品国产麻豆| 成人久久久观看免费毛片|