青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

每天早晨叫醒你的不是鬧鐘,而是夢想

  C++博客 :: 首頁 :: 聯系 :: 聚合  :: 管理
  62 Posts :: 0 Stories :: 5 Comments :: 0 Trackbacks

常用鏈接

留言簿(1)

我參與的團隊

搜索

  •  

最新評論

閱讀排行榜

評論排行榜

 

題目(一):我們可以用static修飾一個類的成員函數,也可以用const修飾類的成員函數(寫在函數的最后表示不能修改成員變量,不是指寫在前面表示返回值為常量)。請問:能不能同時用staticconst修飾類的成員函數?

分析:答案是不可以。C++編譯器在實現const的成員函數的時候為了確保該函數不能修改類的實例的狀態,會在函數中添加一個隱式的參數const this*。但當一個成員為static的時候,該函數是沒有this指針的。也就是說此時static的用法和static是沖突的。

我們也可以這樣理解:兩者的語意是矛盾的。static的作用是表示該函數只作用在類型的靜態變量上,與類的實例沒有關系;而const的作用是確保函數不能修改類的實例的狀態,與類型的靜態變量沒有關系。因此不能同時用它們。

題目(二):運行下面的代碼,輸出是什么?

 1 class A
 2 
 3 {
 4 
 5 };
 6 
 7  
 8 class B
 9 
10 {
11 
12 public:
13 
14         B() {}
15 
16         ~B() {}
17 
18 };
19 
20  
21 class C
22 
23 {
24 
25 public:
26 
27         C() {}
28 
29         virtual ~C() {}
30 
31 };
32 
33  
34 int _tmain(int argc, _TCHAR* argv[])
35 
36 {
37 
38         printf("%d, %d, %d\n"sizeof(A), sizeof(B), sizeof(C));
39 
40         return 0;
41 
42 }
43 
44 


分析:答案是1, 1, 4class A是一個空類型,它的實例不包含任何信息,本來求sizeof應該是0。但當我們聲明該類型的實例的時候,它必須在內存中占有一定的空間,否則無法使用這些實例。至于占用多少內存,由編譯器決定。Visual Studio 2008中每個空類型的實例占用一個byte的空間。

class Bclass A的基礎上添加了構造函數和析構函數。由于構造函數和析構函數的調用與類型的實例無關(調用它們只需要知道函數地址即可),在它的實例中不需要增加任何信息。所以sizeof(B)sizeof(A)一樣,在Visual Studio 2008中都是1

class Cclass B的基礎上把析構函數標注為虛擬函數。C++的編譯器一旦發現一個類型中有虛擬函數,就會為該類型生成虛函數表,并在該類型的每一個實例中添加一個指向虛函數表的指針。在32位的機器上,一個指針占4個字節的空間,因此sizeof(C)4

題目(三):運行下面中的代碼,得到的結果是什么

 1 class A
 2 
 3 {
 4 
 5 private:
 6 
 7         int m_value;
 8 
 9  
10 
11 public:
12 
13         A(int value)
14 
15         {
16 
17                 m_value = value;
18 
19         }
20 
21         void Print1()
22 
23         {
24 
25                 printf("hello world");
26 
27         }
28 
29         void Print2()
30 
31         {
32 
33                 printf("%d", m_value);
34 
35         }
36 
37 };
38 
39  
40 
41 int _tmain(int argc, _TCHAR* argv[])
42 
43 {
44 
45         A* pA = NULL;
46 
47         pA->Print1();
48 
49         pA->Print2();
50 
51  
52 
53         return 0;
54 
55 }
56 
57 


分析:答案是Print1調用正常,打印出hello world,但運行至Print2時,程序崩潰。調用Print1時,并不需要pA的地址,因為Print1的函數地址是固定的。編譯器會給Print1傳入一個this指針,該指針為NULL,但在Print1中該this指針并沒有用到。只要程序運行時沒有訪問不該訪問的內存就不會出錯,因此運行正常。在運行print2時,需要this指針才能得到m_value的值。由于此時this指針為NULL,因此程序崩潰了。

題目(四):運行下面中的代碼,得到的結果是什么

 1 class A
 2 
 3 {
 4 
 5 private:
 6 
 7         int m_value;
 8 
 9  
10 
11 public:
12 
13         A(int value)
14 
15         {
16 
17                 m_value = value;
18 
19         }
20 
21         void Print1()
22 
23         {
24 
25                 printf("hello world");
26 
27         }
28 
29         virtual void Print2()
30 
31         {
32 
33                 printf("hello world");
34 
35         }
36 
37 };
38 
39  
40 
41 int _tmain(int argc, _TCHAR* argv[])
42 
43 {
44 
45         A* pA = NULL;
46 
47         pA->Print1();
48 
49         pA->Print2();
50 
51  
52 
53         return 0;
54 
55 }
56 
57 


分析:答案是Print1調用正常,打印出hello world,但運行至Print2時,程序崩潰。Print1的調用情況和上面的題目一樣,不在贅述。由于Print2是虛函數。C++調用虛函數的時候,要根據實例(即this指針指向的實例)中虛函數表指針得到虛函數表,再從虛函數表中找到函數的地址。由于這一步需要訪問實例的地址(即this指針),而此時this指針為空指針,因此導致內存訪問出錯。

題目(五):靜態成員函數能不能同時也是虛函數

分析:答案是不能。調用靜態成員函數不要實例。但調用虛函數需要從一個實例中指向虛函數表的指針以得到函數的地址,因此調用虛函數需要一個實例。兩者相互矛盾。


題目(六):運行下列C++代碼,輸出什么?
struct Point3D
{
        
int x;
        
int y;
        
int z;
}
;


int _tmain(int argc, _TCHAR* argv[])
{
        Point3D
* pPoint = NULL;
        
int offset = (int)(&(pPoint)->z);
        printf(
"%d", offset);
        
return 0;

}



答案:輸出8。由于在pPoint->z的前面加上了取地址符號,運行到此時的時候,會在pPoint的指針地址上加z在類型Point3D中的偏移量8。由于pPoint的地址是0,因此最終offset的值是8

&(pPoint->z)的語意是求pPoint中變量z的地址(pPoint的地址0z的偏移量8),并不需要訪問pPoint指向的內存。只要不訪問非法的內存,程序就不會出錯。

題目(七):運行下列C++代碼,輸出什么?

class A

{

public:

        A()

        {

                Print();

        }

        
virtual void Print()

        {

                printf(
"A is constructed.\n");

        }

};

class B: public A
{
public:
        B()
        {
                Print();
        }
        
virtual void Print()
        {
                printf(
"B is constructed.\n");
        }
};

int _tmain(int argc, _TCHAR* argv[])
{
        A
* pA = new B();
        delete pA;
        
return 0;

}


答案:先后打印出兩行:A is constructed. B is constructed. 調用B的構造函數時,先會調用B的基類及A的構造函數。然后在A的構造函數里調用Print。由于此時實例的類型B的部分還沒有構造好,本質上它只是A的一個實例,他的虛函數表指針指向的是類型A的虛函數表。因此此時調用的PrintA::Print,而不是B::Print。接著調用類型B的構造函數,并調用Print。此時已經開始構造B,因此此時調用的PrintB::Print

同樣是調用虛擬函數Print,我們發現在類型A的構造函數中,調用的是A::Print,在B的構造函數中,調用的是B::Print。因此虛函數在構造函數中,已經失去了虛函數的動態綁定特性。

題目(12:運行下圖中的C++代碼,輸出是什么

#include <iostream>

 

class A

{

private:

        int n1;

        int n2;

public:

        A(): n2(0), n1(n2 + 2)

        {

        }

 

        void Print()

        {

                std::cout << "n1: " << n1 << ", n2: " << n2 << std::endl;

        }

};

 

int _tmain(int argc, _TCHAR* argv[])

{

        A a;

        a.Print();

 

        return 0;

}

答案:輸出n1是一個隨機的數字,n20。在C++中,成員變量的初始化順序與變量在類型中的申明順序相同,而與它們在構造函數的初始化列表中的順序無關。因此在這道題中,會首先初始化n1,而初始n1的參數n2還沒有初始化,是一個隨機值,因此n1就是一個隨機值。初始化n2時,根據參數0對其初始化,故n2=0

題目(13編譯運行下圖中的C++代碼,結果是什么?(A)編譯錯誤;(B)編譯成功,運行時程序崩潰;(C)編譯運行正常,輸出10。請選擇正確答案并分析原因

#include <iostream>

 

class A

{

private:

        int value;

 

public:

        A(int n)

        {

                value = n;

        }

 

        A(A other)

        {

                value = other.value;

        }

 

        void Print()

        {

                std::cout << value << std::endl;

        }

};

 

int _tmain(int argc, _TCHAR* argv[])

{

        A a = 10;

        A b = a;

        b.Print();

 

        return 0;

}

答案:編譯錯誤。在復制構造函數中傳入的參數是A的一個實例。由于是傳值,把形參拷貝到實參會調用復制構造函數。因此如果允許復制構造函數傳值,那么會形成永無休止的遞歸并造成棧溢出。因此C++的標準不允許復制構造函數傳值參數,而必須是傳引用或者常量引用。在Visual StudioGCC中,都將編譯出錯。

題目(14運行下圖中的C++代碼,輸出是什么

int SizeOf(char pString[])

{

        return sizeof(pString);

}

 

int _tmain(int argc, _TCHAR* argv[])

{

        char* pString1 = "google";

        int size1 = sizeof(pString1);

        int size2 = sizeof(*pString1);

 

        char pString2[100] = "google";

        int size3 = sizeof(pString2);

        int size4 = SizeOf(pString2);

 

        printf("%d, %d, %d, %d", size1, size2, size3, size4);

 

        return 0;

}

答案4, 1, 100, 4pString1是一個指針。在32位機器上,任意指針都占4個字節的空間。*pString1是字符串pString1的第一個字符。一個字符占一個字節。pString2是一個數組,sizeof(pString2)是求數組的大小。這個數組包含100個字符,因此大小是100個字節。而在函數SizeOf中,雖然傳入的參數是一個字符數組,當數組作為函數的參數進行傳遞時,數組就自動退化為同類型的指針。因此size4也是一個指針的大小,為4.

題目(15:運行下圖中代碼,輸出的結果是什么?這段代碼有什么問題

#include <iostream>

 

class A

{

public:

        A()

        {

                std::cout << "A is created." << std::endl;

        }

 

        ~A()

        {

                std::cout << "A is deleted." << std::endl;

        }

};

 

class B : public A

{

public:

        B()

        {

                std::cout << "B is created." << std::endl;

        }

 

        ~B()

        {

                std::cout << "B is deleted." << std::endl;

        }

};

 

int _tmain(int argc, _TCHAR* argv[])

{

        A* pA = new B();

        delete pA;

 

        return 0;

}

答案:輸出三行,分別是:A is created. B is created. A is deleted。用new創建B時,回調用B的構造函數。在調用B的構造函數的時候,會先調用A的構造函數。因此先輸出A is created. B is created.

接下來運行delete語句時,會調用析構函數。由于pA被聲明成類型A的指針,同時基類A的析構函數沒有標上virtual,因此只有A的析構函數被調用到,而不會調用B的析構函數。

由于pA實際上是指向一個B的實例的指針,但在析構的時候只調用了基類A的析構函數,卻沒有調用B的析構函數。這就是一個問題。如果在類型B中創建了一些資源,比如文件句柄、內存等,在這種情況下都得不到釋放,從而導致資源泄漏。

問題(16:運行如下的C++代碼,輸出是什么?

class A

{

public:

    virtual void Fun(int number = 10)

    {

        std::cout << "A::Fun with number " << number;

    }

};

 

class B: public A

{

public:

    virtual void Fun(int number = 20)

    {

        std::cout << "B::Fun with number " << number;

    }

};

 

int main()

{

    B b;

    A &a = b;

    a.Fun();

}

答案輸出B::Fun with number 10。由于a是一個指向B實例的引用,因此在運行的時候會調用B::Fun。但缺省參數是在編譯期決定的。在編譯的時候,編譯器只知道a是一個類型a的引用,具體指向什么類型在編譯期是不能確定的,因此會按照A::Fun的聲明把缺省參數number設為10

            這一題的關鍵在于理解確定缺省參數的值是在編譯的時候,但確定引用、指針的虛函數調用哪個類型的函數是在運行的時候。

問題(16:運行如下的C++代碼,輸出是什么?

class A

{

public:

    virtual void Fun(int number = 10)

    {

        std::cout << "A::Fun with number " << number;

    }

};

 

class B: public A

{

public:

    virtual void Fun(int number = 20)

    {

        std::cout << "B::Fun with number " << number;

    }

};

 

int main()

{

    B b;

    A &a = b;

    a.Fun();

}

答案輸出B::Fun with number 10。由于a是一個指向B實例的引用,因此在運行的時候會調用B::Fun。但缺省參數是在編譯期決定的。在編譯的時候,編譯器只知道a是一個類型a的引用,具體指向什么類型在編譯期是不能確定的,因此會按照A::Fun的聲明把缺省參數number設為10

            這一題的關鍵在于理解確定缺省參數的值是在編譯的時候,但確定引用、指針的虛函數調用哪個類型的函數是在運行的時候。

問題(19運行下圖中C代碼,輸出的結果是什么

int _tmain(int argc, _TCHAR* argv[])

{

    char str1[] = "hello world";

    char str2[] = "hello world";

 

    char* str3 = "hello world";

    char* str4 = "hello world";

 

    if(str1 == str2)

        printf("str1 and str2 are same.\n");

    else

        printf("str1 and str2 are not same.\n");

 

    if(str3 == str4)

        printf("str3 and str4 are same.\n");

    else

        printf("str3 and str4 are not same.\n");

 

    return 0;

}

答案輸出兩行。第一行是str1 and str2 are not same,第二行是str3 and str4 are same

str1str2是兩個字符串數組。我們會為它們分配兩個長度為12個字節的空間,并把"hello world"的內容分別拷貝到數組中去。這是兩個初始地址不同的數組,因此比較str1str2的值,會不相同str3str4是兩個指針,我們無需為它們分配內存以存儲字符串的內容,而只需要把它們指向"hello world“在內存中的地址就可以了。由于"hello world”是常量字符串,它在內存中只有一個拷貝,因此str3str4指向的是同一個地址。因此比較str3str4的值,會是相同的

posted on 2011-04-29 13:44 沛沛 閱讀(740) 評論(0)  編輯 收藏 引用 所屬分類: C++
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            美女91精品| 在线看片日韩| 亚洲乱码日产精品bd| 久久精品综合一区| 亚洲摸下面视频| 在线综合+亚洲+欧美中文字幕| 亚洲精品视频在线| 亚洲精品一区在线观看香蕉| 亚洲人成高清| 中文网丁香综合网| 亚洲综合电影一区二区三区| 亚洲男人第一av网站| 宅男噜噜噜66一区二区| 亚洲一区区二区| 午夜视频一区在线观看| 久久成人亚洲| 欧美国产精品v| 日韩午夜在线电影| 亚洲精品在线免费| 一本久道久久综合婷婷鲸鱼| 一区二区久久| 亚洲一级在线观看| 久久精品亚洲一区二区三区浴池| 久久久水蜜桃av免费网站| 久久亚洲一区| 亚洲人屁股眼子交8| 亚洲日本一区二区三区| 一区二区精品| 欧美综合二区| 久久精品1区| 久久蜜桃精品| 久久噜噜噜精品国产亚洲综合| 91久久精品国产| 亚洲国产精品热久久| 9l国产精品久久久久麻豆| 欧美午夜一区| 欧美日韩中文精品| 久久综合一区二区| 99ri日韩精品视频| 欧美一区二区三区在线免费观看| 蜜乳av另类精品一区二区| 欧美日韩 国产精品| 国产精品自拍小视频| 亚洲第一福利在线观看| 亚洲欧美日韩在线一区| 亚洲高清免费视频| 亚洲影音先锋| 欧美精品一区二区在线播放| 国产一区视频在线看| 亚洲私拍自拍| 欧美国产一区在线| 99pao成人国产永久免费视频| 亚洲私人影院| 亚洲尤物影院| 亚洲欧洲精品一区二区三区| 欧美一二三区精品| 国产精品成人一区二区网站软件 | 欧美精品乱码久久久久久按摩| 国产日韩欧美一二三区| 亚洲视频1区2区| 欧美激情bt| 欧美.www| 亚洲国产美女| 免费日韩精品中文字幕视频在线| 欧美一区二区三区四区视频| 国产精品素人视频| 亚洲欧美中文另类| 国产精品99久久久久久久久 | 亚洲精品九九| 欧美—级高清免费播放| 亚洲乱码国产乱码精品精98午夜| 欧美成人黑人xx视频免费观看| 久久成人一区二区| 欧美视频在线视频| 欧美国产欧美亚洲国产日韩mv天天看完整 | 久久伊人免费视频| 日韩视频中文| 国产日韩成人精品| 久久久久久999| 久久精品国产免费| 久久免费视频在线观看| 国产日韩一区二区| 亚洲免费电影在线| 亚洲国产女人aaa毛片在线| 亚洲国产va精品久久久不卡综合| 免费永久网站黄欧美| 亚洲一区二区三区色| 久久在线91| 欧美日韩在线播| 久久看片网站| 欧美另类变人与禽xxxxx| 亚洲亚洲精品三区日韩精品在线视频| 怡红院av一区二区三区| 男男成人高潮片免费网站| 久久精品30| 欧美一区二区三区四区视频| 欧美有码在线视频| 一区二区三区视频在线看 | 亚洲视频在线视频| 亚洲私人黄色宅男| 一区二区三区欧美| 久久久综合免费视频| 亚洲欧洲一区二区在线播放| 国产精品多人| 欧美日产在线观看| 日韩一级在线观看| 亚洲日本欧美天堂| 欧美高清视频免费观看| 欧美有码视频| 欧美亚洲日本网站| 亚洲一区综合| 亚洲午夜久久久| 亚洲免费播放| 亚洲日本成人网| 亚洲国产第一| 亚洲日本中文| 亚洲第一毛片| 亚洲国产成人久久综合一区| 欧美三级第一页| 欧美激情二区三区| 欧美一区二区三区在线视频| 久久爱www久久做| 一区二区三区蜜桃网| 久久精品国产999大香线蕉| 亚洲视频一区二区| 久久综合九色欧美综合狠狠| 校园激情久久| 欧美日韩另类综合| 亚洲大片在线| 亚洲高清在线观看一区| 亚洲免费小视频| 亚洲色图制服丝袜| 欧美成人精品激情在线观看| 久久琪琪电影院| 亚洲深夜av| 亚洲精品久久久久久一区二区| 亚洲欧美久久久| 亚洲欧美文学| 欧美小视频在线观看| 91久久久久久久久| 亚洲欧洲美洲综合色网| 久久精品72免费观看| 国产精品美女久久久久久久| 一区二区冒白浆视频| 久久精品首页| 久久人人97超碰人人澡爱香蕉 | 欧美国产精品日韩| 久久综合国产精品| 伊人久久综合| 久久久久九九九九| 欧美11—12娇小xxxx| 在线观看国产精品淫| 久久久噜噜噜久噜久久| 欧美福利精品| 99热这里只有成人精品国产| 欧美激情视频一区二区三区不卡| 欧美激情欧美激情在线五月| 亚洲精品影院| 国产精品欧美日韩一区二区| 亚洲欧美在线x视频| 久久久91精品国产一区二区精品| 狠狠色丁香久久婷婷综合丁香| 老司机一区二区三区| 亚洲免费观看高清在线观看| 亚洲免费在线| 韩日在线一区| 欧美国产欧美亚洲国产日韩mv天天看完整| 91久久在线| 欧美一区二区三区成人| 伊人久久大香线| 欧美精品九九| 午夜精品视频一区| 亚洲第一福利视频| 亚洲欧美文学| 亚洲黄色高清| 国产精品都在这里| 久久精品视频导航| 亚洲久久一区| 久久久精品国产免费观看同学| 亚洲国产视频一区二区| 国产精品久久久久久久久久直播| 性色一区二区| 亚洲欧洲久久| 久久天堂成人| 亚洲专区一区二区三区| 伊人成人在线| 国产精品久久久久国产a级| 久久久最新网址| 午夜精品亚洲| 日韩视频一区| 免费在线观看日韩欧美| 这里只有精品视频在线| 黄色成人精品网站| 欧美日韩dvd在线观看| 久久手机精品视频| 一区二区三区久久久| 国内一区二区在线视频观看| 欧美精品成人一区二区在线观看 | 韩日欧美一区二区| 欧美日韩一区国产|