C++深入淺出講解隱藏this指針的用法
本篇文章我們將一起討論在有趣的知識(shí)點(diǎn)--隱藏的this指針。本篇我們要使用到之前我們所學(xué)習(xí)到的C++類與對(duì)象,如果有各位小伙伴還不曾了解類與對(duì)象的簡(jiǎn)單思想,可以訪問(wèn)上篇:C++深入講解類與對(duì)象之OOP面向?qū)ο缶幊膛c封裝
在之后的學(xué)習(xí)中,我們將認(rèn)識(shí)一個(gè)新的類:日期類Date。正如我們所想的那樣,傳入一個(gè)日期,我們可以輸出我們所輸入的日期。
1.this指針的引出
那我們首先來(lái)看一下,這段代碼會(huì)輸出什么結(jié)果呢?
class Date
{
public:
void Display()
{
cout << _year << "-" << _month << "-" << _day << endl;
}
void SetDate(int year, int month, int day)
{
_year = year;
_month = month;
_day = day;
}
private:
int _year; // 年
int _month; // 月
int _day; // 日
};
int main()
{
Date d1, d2;
d1.SetDate(2022, 5, 11);
d2.SetDate(2022, 5, 12);
d1.Display();
d2.Display();
return 0;
}輸出結(jié)果:

我們首先可以通過(guò)匯編來(lái)看看,d1,d2調(diào)用的函數(shù)是否相同。

我們可以發(fā)現(xiàn),最終打印的時(shí)候調(diào)用的Display()是同一個(gè)函數(shù),那么既然d1,d2調(diào)用的都是同一個(gè)函數(shù),編譯器如何知道d1是2022-5-11,d2是2022-5-12呢?Display()都訪問(wèn)的_year,_month,_day。而且去公共代碼區(qū)訪問(wèn)的Display(),這是為什么呢?
這是因?yàn)镃++在這段代碼中做出手腳,C++在這里增加了一個(gè)this指針,這里是因?yàn)镈isplay會(huì)增加一個(gè)this形參。C++編譯器給每個(gè)“非靜態(tài)的成員函數(shù)“增加了一個(gè)隱藏的指針參數(shù),讓該指針指向當(dāng)前對(duì)象(函數(shù)運(yùn)行時(shí)調(diào)用該函數(shù)的對(duì)象),在函數(shù)體中所有成員變量的操作,都是通過(guò)該指針去訪問(wèn)。只不過(guò)所有的操作對(duì)用戶是透明的,即用戶不需要來(lái)傳遞,編譯器自動(dòng)完成。
在調(diào)用的時(shí)候也傳的是各自的地址。這樣就十分清晰明了了。這就是隱含的this指針




注意:我們不能顯示的寫(xiě)出來(lái),因?yàn)樗请[含的,我們不能搶了編譯器的活。但是我們可以直接在類里面用。
2.this指針的特性
在真正的編譯器中this指針的用const修飾的,this指針本身是不能被修改的,但是內(nèi)容是可以修改的。并且我們是可以使用的

我們可以在類中打印一下this指針,并且我們?cè)谝餐瑫r(shí)打印一下d1和d2的地址,我們來(lái)看一下:
class Date
{
public:
void Display()
{
//使用this指針
cout << this << endl;
cout << _year << "-" << _month << "-" << _day << endl;
}
void SetDate(int year, int month, int day)
{
_year = year;
_month = month;
_day = day;
}
private:
int _year; // 年
int _month; // 月
int _day; // 日
};
int main()
{
Date d1, d2;
cout <<"d1:"<< & d1 << endl;
cout <<"d2:"<< & d2 << endl;
d1.SetDate(2022, 5, 11);
d2.SetDate(2022, 5, 12);
d1.Display();
d2.Display();
return 0;
}運(yùn)行結(jié)果:

并且我們還能這樣寫(xiě),但是我們不能顯示的寫(xiě)出Date* this。

我們接下來(lái)再看看this指針是不能修改的,大家看下面這個(gè)能過(guò)嗎?答案肯定是不能的,因?yàn)閠his是被const修飾的,不能修改this指針的。

我們會(huì)發(fā)現(xiàn) 編譯器也會(huì)報(bào)錯(cuò):error C2106: “=”: 左操作數(shù)必須為左值

this指針的特性總結(jié):
1. this指針的類型:類類型* const。
2. 只能在“成員函數(shù)”的內(nèi)部使用。
3. this指針本質(zhì)上其實(shí)是一個(gè)成員函數(shù)的形參,是對(duì)象調(diào)用成員函數(shù)時(shí),將對(duì)象地址作為實(shí)參傳遞給this形參。所以對(duì)象中不存儲(chǔ)this指針。
4. this指針是成員函數(shù)第一個(gè)隱含的指針形參,一般情況由編譯器通過(guò)ecx寄存器自動(dòng)傳遞,不需要用戶傳遞。
3.練習(xí)一下
(1)下面的程序的運(yùn)行結(jié)果是? A.編譯報(bào)錯(cuò) B.運(yùn)行崩潰 C.正常運(yùn)行
class A
{
public:
void Show()
{
cout << "show()" << endl;
}
private:
int _a;
};
int main()
{
A* p = nullptr;
p->Show();
return 0;
}結(jié)果:C
原因:Show()函數(shù)是存在公共代碼區(qū)中,編譯的時(shí)候在公共代碼區(qū)中找到這個(gè)函數(shù),和普通的函數(shù)調(diào)用是一樣的,只需要call函數(shù)地址就行。我們發(fā)現(xiàn)這里p是空指針,傳過(guò)去的this指針只是接收了p的空指針,就類似于this指針被初始化為空指針。這是允許的。


(2)下面的程序的運(yùn)行結(jié)果是? A.編譯報(bào)錯(cuò) B.運(yùn)行崩潰 C.正常運(yùn)行
class B
{
public:
void PrintA()
{
cout << _a << endl;
}
private:
int _a;
};
int main()
{
B* p2 = nullptr;
p2->PrintA();
return 0;
}結(jié)果:B
原因:此程序崩潰是在PrintA()中,會(huì)隱含一個(gè)this->_a,而this指針是一個(gè)空指針,訪問(wèn)this指針_a的位置,就要對(duì)空指針進(jìn)行解引用,此時(shí)就會(huì)崩潰。我們也可通過(guò)調(diào)試觀察到。


(3)this指針是存在哪里的?
a.棧 b.堆 c.靜態(tài)區(qū) d.常量區(qū)
答案:a
解釋:this指針是個(gè)形參,形參是在函數(shù)的棧楨里,在函數(shù)的棧楨里面的變量是屬于棧中的。
有時(shí)編譯器會(huì)使用寄存器對(duì)其進(jìn)行優(yōu)化,this指針會(huì)存在寄存器中。


到此這篇關(guān)于C++深入淺出講解隱藏this指針的用法的文章就介紹到這了,更多相關(guān)C++this指針內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C語(yǔ)言實(shí)現(xiàn)通訊錄功能的流程與代碼
通訊錄是一個(gè)可以記錄親人、好友信息的工具,這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言實(shí)現(xiàn)通訊錄管理,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-06-06
基于Matlab實(shí)現(xiàn)野狗優(yōu)化算法的示例代碼
野狗優(yōu)化算法(Dingo?Optimization?Algorithm,?DOA)模仿澳大利亞野狗的社交行為。DOA算法的靈感來(lái)源于野狗的狩獵策略,即迫害攻擊、分組策略和食腐行為。本文將通過(guò)Matlab實(shí)現(xiàn)這一算法,感興趣的可以了解一下2022-04-04
C++中使用哈希表(unordered_map)的一些常用操作方法
C++標(biāo)準(zhǔn)庫(kù)中使用的unordered_map底層實(shí)現(xiàn)是哈希表,下面這篇文章主要給大家介紹了關(guān)于C++中使用哈希表(unordered_map)的一些常用操作方法,需要的朋友可以參考下2022-03-03
C語(yǔ)言編程C++編輯器及調(diào)試工具操作命令詳解
這篇文章主要介紹了C語(yǔ)言編程C++編輯調(diào)試工具操作命令詳解,本文章對(duì)C++調(diào)試工具的命令操作進(jìn)行了詳細(xì)的講解,有需要的朋友可以借鑒參考下2021-09-09
C語(yǔ)言冷門(mén)知識(shí)之你可能沒(méi)聽(tīng)過(guò)的柔性數(shù)組
柔性數(shù)組(Flexible Array)是引入的一個(gè)新特性,它允許你在定義結(jié)構(gòu)體時(shí)創(chuàng)建一個(gè)空數(shù)組,而這個(gè)數(shù)組的大小可以在程序運(yùn)行的過(guò)程中根據(jù)你的需求進(jìn)行更改特別注意的一點(diǎn)是:這個(gè)空數(shù)組必須聲明為結(jié)構(gòu)體的最后一個(gè)成員,并且還要求這樣的結(jié)構(gòu)體至少包含一個(gè)其他類型的成員2021-10-10
C語(yǔ)言實(shí)現(xiàn)餐廳管理系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言實(shí)現(xiàn)餐廳管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-06-06
C語(yǔ)言的進(jìn)制轉(zhuǎn)換及算法實(shí)現(xiàn)教程
這篇文章主要介紹了C語(yǔ)言的進(jìn)制轉(zhuǎn)換及算法實(shí)現(xiàn)的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-01-01

