題目(一):我們可以用static修飾一個類的成員函數(shù),也可以用const修飾類的成員函數(shù)(寫在函數(shù)的最后表示不能修改成員變量,不是指寫在前
分析:答案是不可以。C++編譯器在實現(xiàn)const的成員函數(shù)的時候為了確保該函數(shù)不能修改類的實例的狀態(tài),會在函數(shù)中添加一個隱式的參數(shù)const this*。但當一個成員為static的時候,該函數(shù)是沒有this指針的。也就是說此時static的用法和static是沖突的。
我們也可以這樣理解:兩者的語意是矛盾的。static的作用是表示該函數(shù)只作用在類型的靜態(tài)變量上,與類的實例沒有關系;而const的作用是確保函數(shù)不能修改類的實例的狀態(tài),與類型的靜態(tài)變量沒有關系。因此不能同時用它們。
題目(二):運行下面的代碼,輸出是什么?
class A
{
};
class B
{
public:
B() {}
~B() {}
};
class C
{
public:
C() {}
virtual ~C() {}
};
int _tmain(int argc, _TCHAR* argv[])
{
printf("%d, %d, %d\n", sizeof(A), sizeof(B), sizeof(C));
return 0;
}
分析:答案是1, 1, 4。class A是一個空類型,它的實例不包含任何信息,本來求sizeof應該是0。但當我們聲明該類型的實例的時候,它必須在內(nèi)存中占有一定的空間,否則無法使用這些實例。至于占用多少內(nèi)存,由編譯器決定。Visual Studio 2008中每個空類型的實例占用一個byte的空間。
class B在class A的基礎上添加了構造函數(shù)和析構函數(shù)。由于構造函數(shù)和析構函數(shù)的調(diào)用與類型的實例無關(調(diào)用它們只需要知道函數(shù)地址即可),在它的實例中不需要增加任何信息。所以sizeof(B)和sizeof(A)一樣,在Visual Studio 2008中都是1。
class C在class B的基礎上把析構函數(shù)標注為虛擬函數(shù)。C++的編譯器一旦發(fā)現(xiàn)一個類型中有虛擬函數(shù),就會為該類型生成虛函數(shù)表,并在該類型的每一個實例中添加一個指向虛函數(shù)表的指針。在32位的機器上,一個指針占4個字節(jié)的空間,因此sizeof(C)是4。
題目(三):運行下面中的代碼,得到的結(jié)果是什么?
class A
{
private:
int m_value;
public:
A(int value)
{
m_value = value;
}
void Print1()
{
printf("hello world");
}
void Print2()
{
printf("%d", m_value);
}
};
int _tmain(int argc, _TCHAR* argv[])
{
A* pA = NULL;
pA->Print1();
pA->Print2();
return 0;
}
分析:答案是Print1調(diào)用正常,打印出hello world,但運行至Print2時,程序崩潰。調(diào)用Print1時,并不需要pA的地址,因為Print1的函數(shù)地址是固定的。編譯器會給Print1傳入一個this指針,該指針為NULL,但在Print1中該this指針并沒有用到。只要程序運行時沒有訪問不該訪問的內(nèi)存就不會出錯,因此運行正常。在運行print2時,需要this指針才能得到m_value的值。由于此時this指針為NULL,因此程序崩潰了。
題目(四):運行下面中的代碼,得到的結(jié)果是什么?
class A
{
private:
int m_value;
public:
A(int value)
{
m_value = value;
}
void Print1()
{
printf("hello world");
}
virtual void Print2()
{
printf("hello world");
}
};
int _tmain(int argc, _TCHAR* argv[])
{
A* pA = NULL;
pA->Print1();
pA->Print2();
return 0;
}
分析:答案是Print1調(diào)用正常,打印出hello world,但運行至Print2時,程序崩潰。Print1的調(diào)用情況和上面的題目一樣,不在贅述。由于Print2是虛函數(shù)。C++調(diào)用虛函數(shù)的時候,要根據(jù)實例(即this指針指向的實例)中虛函數(shù)表指針得到虛函數(shù)表,再從虛函數(shù)表中找到函數(shù)的地址。由于這一步需要訪問實例的地址(即this指針),而此時this指針為空指針,因此導致內(nèi)存訪問出錯。
題目(五):靜態(tài)成員函數(shù)能不能同時也是虛函數(shù)?
分析:答案是不能。調(diào)用靜態(tài)成員函數(shù)不要實例。但調(diào)用虛函數(shù)需要從一個實例中指向虛函數(shù)表的指針以得到函數(shù)的地址,因此調(diào)用虛函數(shù)需要一個實例。兩者相互矛盾。