c中this的用法
c中this的用法的用法你知道嗎?下面小編就跟你們詳細(xì)介紹下c中this的用法的用法,希望對你們有用。
c中this的用法的用法如下:
1. this指針的用處:
一個對象的this指針并不是對象本身的一部分,不會影響sizeof(對象)的結(jié)果。this作用域是在類內(nèi)部,當(dāng)在類的非靜態(tài)成員函數(shù)中訪問類的非靜態(tài)成員的時候,編譯器會自動將對象本身的地址作為一個隱含參數(shù)傳遞給函數(shù)。也就是說,即使你沒有寫上this指針,編譯器在編譯的時候也是加上this的,它作為非靜態(tài)成員函數(shù)的隱含形參,對各成員的訪問均通過this進(jìn)行。 例如,調(diào)用date.SetMonth(9) <===> SetMonth(&date, 9),this幫助完成了這一轉(zhuǎn)換 .
2. this指針的使用:
一種情況就是,在類的非靜態(tài)成員函數(shù)中返回類對象本身的時候,直接使用 return *this;另外一種情況是當(dāng)參數(shù)與成員變量名相同時,如this->n = n (不能寫成n = n)。
3. this指針程序示例:
this指針存在于類的成員函數(shù)中,指向被調(diào)用函數(shù)所在的類實例的地址。 根據(jù)以下程序來說明this指針
#include
class Point { int x, y;
public:
Point(int a, int b) { x=a; y=b;}
void MovePoint( int a, int b){ x+=a; y+=b;}
void print(){ cout<<"x="<
};
void main( ) {
Point point1( 10,10);
point1.MovePoint(2,2);
point1.print( );
}
當(dāng)對象point1調(diào)用MovePoint(2,2)函數(shù)時,即將point1對象的地址傳遞給了this指針。
MovePoint函數(shù)的原型應(yīng)該是 void MovePoint( Point *this, int a, int b);第一個參數(shù)是指向該類對象的一個指針,我們在定義成員函數(shù)時沒看見是因為這個參數(shù)在類中是隱含的。這樣point1的地址傳遞給了this,所以在MovePoint函數(shù)中便顯式的寫成:
void MovePoint(int a, int b) { this->x +=a; this-> y+= b;} 即可以知道,point1調(diào)用該函數(shù)后,也就是point1的數(shù)據(jù)成員被調(diào)用并更新了值。 即該函數(shù)過程可寫成 point1.x+= a; point1. y + = b;
4. 關(guān)于this指針的一個經(jīng)典回答:
當(dāng)你進(jìn)入一個房子后,
你可以看見桌子、椅子、地板等,
但是房子你是看不到全貌了。
對于一個類的實例來說,
你可以看到它的成員函數(shù)、成員變量,
但是實例本身呢?
this是一個指針,它時時刻刻指向你這個實例本身
5. 類的this指針有以下特點:
?。?)this只能在成員函數(shù)中使用。
全局函數(shù)、靜態(tài)函數(shù)都不能使用this.
實際上,成員函數(shù)默認(rèn)第一個參數(shù)為T * const this。
如:
class A
{
public:
int func(int p)
{
}
};
其中,func的原型在編譯器看來應(yīng)該是:
int func(A * const this,int p);
(2)由此可見,this在成員函數(shù)的開始前構(gòu)造,在成員函數(shù)的結(jié)束后清除。
這個生命周期同任何一個函數(shù)的參數(shù)是一樣的,沒有任何區(qū)別。
當(dāng)調(diào)用一個類的成員函數(shù)時,編譯器將類的指針作為函數(shù)的this參數(shù)傳遞進(jìn)去。如:
A a;
a.func(10);
此處,編譯器將會編譯成:
A::func(&a,10);
看起來和靜態(tài)函數(shù)沒差別,對嗎?不過,區(qū)別還是有的。編譯器通常會對this指針做一些優(yōu)化,因此,this指針的傳遞效率比較高--如VC通常是通過ecx寄存器傳遞this參數(shù)的。
?。?)幾個this指針的易混問題。
A. this指針是什么時候創(chuàng)建的?
this在成員函數(shù)的開始執(zhí)行前構(gòu)造,在成員的執(zhí)行結(jié)束后清除。
但是如果class或者struct里面沒有方法的話,它們是沒有構(gòu)造函數(shù)的,只能當(dāng)做C的struct使用。采用 TYPE xx的方式定義的話,在棧里分配內(nèi)存,這時候this指針的值就是這塊內(nèi)存的地址。采用new的方式 創(chuàng)建對象的話,在堆里分配內(nèi)存,new操作符通過eax返回分配 的地址,然后設(shè)置給指針變量。之后去調(diào) 用構(gòu)造函數(shù)(如果有構(gòu)造函數(shù)的話),這時將這個內(nèi)存塊的地址傳給ecx,之后構(gòu)造函數(shù)里面怎么處理請 看上面的回答。
B. this指針存放在何處?堆、棧、全局變量,還是其他?
this指針會因編譯器不同而有不同的放置位置??赡苁菞?,也可能是寄存器,甚至全局變量。在匯編級 別里面,一個值只會以3種形式出現(xiàn):立即數(shù)、寄存器值和內(nèi)存變量值。不是存放在寄存器就是存放在內(nèi) 存中,它們并不是和高級語言變量對應(yīng)的。
C. this指針是如何傳遞類中的函數(shù)的?綁定?還是在函數(shù)參數(shù)的首參數(shù)就是this指針?那么,this指針 又是如何找到“類實例后函數(shù)的”?
大多數(shù)編譯器通過ecx寄存器傳遞this指針。事實上,這也是一個潛規(guī)則。一般來說,不同編譯器都會遵從一致的傳參規(guī)則,否則不同編譯器產(chǎn)生的obj就無法匹配了。
在call之前,編譯器會把對應(yīng)的對象地址放到eax中。this是通過函數(shù)參數(shù)的首參來傳遞的。this指針在調(diào)用之前生成,至于“類實例后函數(shù)”,沒有這個說法。類在實例化時,只分配類中的變量空間,并沒有為函數(shù)分配空間。自從類的函數(shù)定義完成后,它就在那兒,不會跑的。
D. this指針是如何訪問類中的變量的?
如果不是類,而是結(jié)構(gòu)體的話,那么,如何通過結(jié)構(gòu)指針來訪問結(jié)構(gòu)中的變量呢?如果你明白這一點的話,就很容易理解這個問題了。
在C++中 ,類和結(jié)構(gòu)是只有一個區(qū)別的:類的成員默認(rèn)是private,而結(jié)構(gòu)是public。
this是類的指針,如果換成結(jié)構(gòu),那this就是結(jié)構(gòu)的指針了。
E. 我們只有獲得一個對象后,才能通過對象使用this指針。如果我們知道一個對象this指針的位置,可以直接使用嗎?
this指針只有在成員函數(shù)中才有定義。因此,你獲得一個對象后,也不能通過對象使用this指針。所以,我們無法知道一個對象的this指針的位置(只有在成員函數(shù)里才有this指針的位置)。當(dāng)然,在成員函數(shù)里,你是可以知道this指針的位置的(可以通過&this獲得),也可以直接使用它。
F. 每個類編譯后,是否創(chuàng)建一個類中函數(shù)表保存函數(shù)指針,以便用來調(diào)用函數(shù)?
普通的類函數(shù)(不論是成員函數(shù),還是靜態(tài)函數(shù))都不會創(chuàng)建一個函數(shù)表來保存函數(shù)指針。只有虛函數(shù)才會被放到函數(shù)表中。但是,即使是虛函數(shù),如果編譯器能明確知道調(diào)用的是哪個函數(shù),編譯器就不會通過函數(shù)表中的指針來間接調(diào)用,而是會直接調(diào)用該函數(shù)。