• <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>

            no_rain

            C++的一個語法問題

            今天看到《C++ primer》上說:可以用派生類對象初始化基類對象,根據是派生類對象可以賦值給基類的對象的引用。這里我覺得有個語義上的問題:派生類在繼承基類的時候,基類的private成員是不可見的,那么基類初始化private對象的時候,就用了派生類的不可見成員去初始化它了。
            先貼段測試代碼:
             1 #include<iostream>
             2 using namespace std;
             3 class A
             4 {
             5       int mem1,mem2;
             6 public:
             7        A(const A& a);
             8        A(int a, int b):mem1(a),mem2(b){}
             9        int fun1(){return mem1;}
            10        int fun2(){return mem2;}
            11 };
            12 A::A(const A& a)
            13 {
            14            mem1 = a.mem1;
            15            mem2 = a.mem2;
            16            cout << "A's copy constructor called"<<endl;
            17 }
            18 class B : public A
            19 {
            20       int mem3;
            21 public:
            22   B(int a, int b, int c): A(a,b),mem3(c){}
            23     int fun3(){return fun1()};
            24 };
            25 int main()
            26 {
            27     B b(1,2,3);
            28     A aob(b);
            29 }
            30 
            這段代碼輸出:A's copy constructor called(這是用G++編譯器的,DEV C++ 編譯通過,可是運行沒有輸出)
            確實如書上說的派生類對象可以賦值給基類的對象的引用,所以調用了拷貝構造函數。其實根據《inside the C++ object model》的說法,派生類的對象中將會保存基類的non-static數據成員的,那么即使不可見,可以用來初始化也在情理之中。
            可是再看被初始化對象調用其成員函數的代碼:
             1 #include<iostream>
             2 using namespace std;
             3 class A
             4 {
             5       int mem1,mem2;
             6 public:
             7        A(const A& a);
             8        A(int a, int b):mem1(a),mem2(b){}
             9        int fun1(){return mem1;}
            10        int fun2(){return mem2;}
            11 };
            12 A::A(const A& a)
            13 {
            14            mem1 = a.mem1;
            15            mem2 = a.mem2;
            16            cout << "A's copy constructor called"<<endl;
            17 }
            18 class B : public A
            19 {
            20       int mem3;
            21 public:
            22   B(int a, int b, int c): A(a,b),mem3(c){}
            23     int fun3(){return fun1()};
            24 };
            25 int main()
            26 {
            27     B b(1,2,3);
            28     A aob(b);
            29     cout <<aob.fun1() << aob.fun2();//the //difference
            30 }
            31 
            這就編譯錯誤了:tess.cpp:28:36: error: request for member ‘fun2’ in ‘aob’, which is of non-class type ‘A(B)’這在兩個上述編譯器都是這樣的結果。那么這個對象就無法調用基類的函數了。
            我個人膚淺的推斷:A(B)將被編譯器認為是一種新的類型對待,那么怎么確定這種類型的接口函數呢?這不就有問題了嗎?
            我再多此一舉的實驗下如下代碼:
             1 #include<iostream>
             2 using namespace std;
             3 class A
             4 {
             5       int mem1,mem2;
             6 public:
             7        A(const A& a);
             8        A(int a, int b):mem1(a),mem2(b){}
             9        int fun1(){return mem1;}
            10        int fun2(){return mem2;}
            11 };
            12 A::A(const A& a)
            13 {
            14            mem1 = a.mem1;
            15            mem2 = a.mem2;
            16            cout << "A's copy constructor called"<<endl;
            17 }
            18 class B : public A
            19 {
            20       int mem3;
            21 public:
            22   B(int a, int b, int c): A(a,b),mem3(c){}
            23     int fun3(){return fun1();}
            24 };
            25 int main()
            26 {
            27     B b(1,2,3);
            28     A aob(b);
            29     cout <<aob.fun3();// the difference
            30 }
            31 
            結果是編譯錯誤:‘class A’ has no member named ‘fun3’這一點也不意外,那么這樣的對象不就真的沒有了接口了?小弟我虛心等待大牛們的解答,希望能在原理上給俺個解釋,不勝感激!

            posted on 2012-03-16 21:13 is-programmer 閱讀(1546) 評論(11)  編輯 收藏 引用

            評論

            # re: C++的一個語法問題 2012-03-16 22:15 遠行

            大哥,你第二個和第三個例子好像有錯誤,class B 里面的fun3,那個;打錯地方了,我試了下都編譯不過了,我改了就可以調用了。。。

            第三個例子的話,這個接口跟對象關系不大吧,接口還是個函數地址,只是用該類對象調用的時候,附帶多傳了個該對象的地址過去,class A的對象是絕對不能在編譯層次上調用class B中的函數的  回復  更多評論   

            # re: C++的一個語法問題 2012-03-16 23:29 is-programmer

            @遠行
            謝謝你的回復!
            我第三個例子的fun3()的分號應該是放在括號內的~這個我改過來了。可是還是
            tess.cpp:29:16: error: ‘class A’ has no member named ‘fun3’(g++ (GCC) 4.6.2);
            我想說的是A(B)即由派生類對象初始化的基類對象為什么就這么不倫不類,A的函數不能調用,B的函數不能調用?  回復  更多評論   

            # re: C++的一個語法問題 2012-03-16 23:46 遠行

            怎么說了,子類可以初始化基類,但是無論怎么樣基類對象不可以調用子類的方法,至于為什么,inside c++ model中說的很清楚,
            從設計意義上講子類對象一定是一個基類對象,所以c++設計成子類默認可以初始化基類,但是方法和數據很不相同吧,函數畢竟只是個代碼的地址,類的成員函數也只是多了個this指針參數,基類對象調用子類中的方法一點意義都沒有,
            如果真想這樣,多態就是這樣做的,只是方法必須是虛函數,而且要用引用或者指針去調用@is-programmer
              回復  更多評論   

            # re: C++的一個語法問題 2012-03-17 09:27 tb

            調試不過啊   回復  更多評論   

            # re: C++的一個語法問題 2012-03-17 12:09 izualzhy

            第二個為什么我的可以通過?結果跟你不一樣?
            zhaoyu@ubuntu:~$ ./test
            A's copy constructor called
            12zhaoyu@ubuntu:~$ g++ -v
            Using built-in specs.
            Target: i486-linux-gnu
            Configured with: ../src/configure -v --with-pkgversion='Ubuntu 4.4.3-4ubuntu5.1' --with-bugurl=file:///usr/share/doc/gcc-4.4/README.Bugs --enable-languages=c,c++,fortran,objc,obj-c++ --prefix=/usr --enable-shared --enable-multiarch --enable-linker-build-id --with-system-zlib --libexecdir=/usr/lib --without-included-gettext --enable-threads=posix --with-gxx-include-dir=/usr/include/c++/4.4 --program-suffix=-4.4 --enable-nls --enable-clocale=gnu --enable-libstdcxx-debug --enable-plugin --enable-objc-gc --enable-targets=all --disable-werror --with-arch-32=i486 --with-tune=generic --enable-checking=release --build=i486-linux-gnu --host=i486-linux-gnu --target=i486-linux-gnu
            Thread model: posix
            gcc version 4.4.3 (Ubuntu 4.4.3-4ubuntu5.1)   回復  更多評論   

            # re: C++的一個語法問題 2012-03-17 13:30 路過

            @is-programmer

            編譯器不對吧,eclipse + mingw gcc4.6.1,aob.func1() 和 aob.func2() 都能正確編譯、執行;aob.func3() 顯然不能決議,沒有這個成員函數。  回復  更多評論   

            # re: C++的一個語法問題 2012-03-17 13:30 is-programmer

            @遠行
            謝謝!你說的這個我明白,但是問題是用子類對象初始化的基類對象,它連自己(基類自己的函數)調用都會編譯錯誤呢?  回復  更多評論   

            # re: C++的一個語法問題 2012-03-17 13:42 is-programmer

            @izualzhy
            謝謝!
            確實是編譯器的問題~這個是在DEV C++的編譯器下才會出現的問題。。。唉DEV C++ 是很久以前開發的,都很久沒有跟新了,所以~~~
            對不起各位,在G++下是可以正常編譯,運行也是正常的,(除了測試代碼3,本身就是錯的)。  回復  更多評論   

            # re: C++的一個語法問題 2012-03-17 13:48 is-programmer

            @izualzhy
            g++ (GCC) 4.6.2 20111027 (Red Hat 4.6.2-1)
            Copyright (C) 2011 Free Software Foundation, Inc.
            你的編譯器可以去跟新了,呵呵。我也編譯過了,昨天忘記是什么情況了。DEV——C++編譯不過  回復  更多評論   

            # re: C++的一個語法問題 2012-03-17 14:16 遠行

            自己的函數調用不可能出錯吧,你去試一下,你這里是調用子類的函數才編譯不過,而且是在靜態調用@is-programmer
              回復  更多評論   

            # re: C++的一個語法問題 2012-03-19 09:57 mendynew

            所謂的”不可見“特性是指在B以及其對象中無法直接操作A中的私有成員,但是通過拷貝構造函數B的對象向上轉型成A,A的拷貝構造函數里自然能操作了。
            有一點一定要記住,你的這幾個例子都是在編譯器決定的,C++的編譯器實現了該語法特性。  回復  更多評論   

            導航

            <2025年5月>
            27282930123
            45678910
            11121314151617
            18192021222324
            25262728293031
            1234567

            統計

            常用鏈接

            留言簿

            隨筆檔案

            文章分類

            文章檔案

            搜索

            最新評論

            閱讀排行榜

            評論排行榜

            精品久久一区二区三区| 国产精品毛片久久久久久久 | 久久av无码专区亚洲av桃花岛| 韩国免费A级毛片久久| 国产韩国精品一区二区三区久久| 2022年国产精品久久久久| 国产农村妇女毛片精品久久| 久久精品卫校国产小美女| 精品久久无码中文字幕| 久久精品国产亚洲Aⅴ香蕉| 久久久久久久97| 久久午夜无码鲁丝片午夜精品| 97久久超碰国产精品2021| 精品久久久久久久久久中文字幕| 久久国产精品-久久精品| 亚洲精品tv久久久久| 99久久精品国产一区二区蜜芽| 国产农村妇女毛片精品久久| 久久国产色av免费看| 韩国免费A级毛片久久| 久久影视综合亚洲| 久久国产免费观看精品| 久久精品国产AV一区二区三区| 996久久国产精品线观看| 久久精品国产男包| 狠狠人妻久久久久久综合蜜桃| 久久99精品久久久久久hb无码 | 久久久www免费人成精品| 久久久91人妻无码精品蜜桃HD| 国产精品久久久久国产A级| 亚洲欧美一区二区三区久久| 色婷婷狠狠久久综合五月| 日韩久久久久中文字幕人妻| 日本免费久久久久久久网站| 国产成人久久精品麻豆一区| 国内精品久久国产大陆| 亚洲色婷婷综合久久| 久久精品国产亚洲AV无码娇色| 久久免费看黄a级毛片| 久久人妻AV中文字幕| 久久精品日日躁夜夜躁欧美|