題目(一):我們可以用static修飾一個(gè)類的成員函數(shù),也可以用const修飾類的成員函數(shù)(寫在函數(shù)的最后表示不能修改成員變量,不是指寫在前
分析:答案是不可以。C++編譯器在實(shí)現(xiàn)const的成員函數(shù)的時(shí)候?yàn)榱舜_保該函數(shù)不能修改類的實(shí)例的狀態(tài),會(huì)在函數(shù)中添加一個(gè)隱式的參數(shù)const this*。但當(dāng)一個(gè)成員為static的時(shí)候,該函數(shù)是沒有this指針的。也就是說(shuō)此時(shí)static的用法和static是沖突的。
我們也可以這樣理解:兩者的語(yǔ)意是矛盾的。static的作用是表示該函數(shù)只作用在類型的靜態(tài)變量上,與類的實(shí)例沒有關(guān)系;而const的作用是確保函數(shù)不能修改類的實(shí)例的狀態(tài),與類型的靜態(tài)變量沒有關(guān)系。因此不能同時(shí)用它們。
題目(二):運(yùn)行下面的代碼,輸出是什么?
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是一個(gè)空類型,它的實(shí)例不包含任何信息,本來(lái)求sizeof應(yīng)該是0。但當(dāng)我們聲明該類型的實(shí)例的時(shí)候,它必須在內(nèi)存中占有一定的空間,否則無(wú)法使用這些實(shí)例。至于占用多少內(nèi)存,由編譯器決定。Visual Studio 2008中每個(gè)空類型的實(shí)例占用一個(gè)byte的空間。
class B在class A的基礎(chǔ)上添加了構(gòu)造函數(shù)和析構(gòu)函數(shù)。由于構(gòu)造函數(shù)和析構(gòu)函數(shù)的調(diào)用與類型的實(shí)例無(wú)關(guān)(調(diào)用它們只需要知道函數(shù)地址即可),在它的實(shí)例中不需要增加任何信息。所以sizeof(B)和sizeof(A)一樣,在Visual Studio 2008中都是1。
class C在class B的基礎(chǔ)上把析構(gòu)函數(shù)標(biāo)注為虛擬函數(shù)。C++的編譯器一旦發(fā)現(xiàn)一個(gè)類型中有虛擬函數(shù),就會(huì)為該類型生成虛函數(shù)表,并在該類型的每一個(gè)實(shí)例中添加一個(gè)指向虛函數(shù)表的指針。在32位的機(jī)器上,一個(gè)指針占4個(gè)字節(jié)的空間,因此sizeof(C)是4。
題目(三):運(yùn)行下面中的代碼,得到的結(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,但運(yùn)行至Print2時(shí),程序崩潰。調(diào)用Print1時(shí),并不需要pA的地址,因?yàn)?/span>Print1的函數(shù)地址是固定的。編譯器會(huì)給Print1傳入一個(gè)this指針,該指針為NULL,但在Print1中該this指針并沒有用到。只要程序運(yùn)行時(shí)沒有訪問(wèn)不該訪問(wèn)的內(nèi)存就不會(huì)出錯(cuò),因此運(yùn)行正常。在運(yùn)行print2時(shí),需要this指針才能得到m_value的值。由于此時(shí)this指針為NULL,因此程序崩潰了。
題目(四):運(yùn)行下面中的代碼,得到的結(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,但運(yùn)行至Print2時(shí),程序崩潰。Print1的調(diào)用情況和上面的題目一樣,不在贅述。由于Print2是虛函數(shù)。C++調(diào)用虛函數(shù)的時(shí)候,要根據(jù)實(shí)例(即this指針指向的實(shí)例)中虛函數(shù)表指針得到虛函數(shù)表,再?gòu)奶摵瘮?shù)表中找到函數(shù)的地址。由于這一步需要訪問(wèn)實(shí)例的地址(即this指針),而此時(shí)this指針為空指針,因此導(dǎo)致內(nèi)存訪問(wèn)出錯(cuò)。
題目(五):靜態(tài)成員函數(shù)能不能同時(shí)也是虛函數(shù)?
分析:答案是不能。調(diào)用靜態(tài)成員函數(shù)不要實(shí)例。但調(diào)用虛函數(shù)需要從一個(gè)實(shí)例中指向虛函數(shù)表的指針以得到函數(shù)的地址,因此調(diào)用虛函數(shù)需要一個(gè)實(shí)例。兩者相互矛盾。
Python交流群
635448130點(diǎn)擊加入群聊UI設(shè)計(jì)交流群
579150876點(diǎn)擊加入群聊Unity交流群
495609038點(diǎn)擊加入群聊HTML5交流群
645591648點(diǎn)擊加入群聊