天天看點

C++:19---this指針

一、this指針介紹

  • 概念:this指針是成員函數的一個隐式參數,在類中本質上就是對象的指針(常量指針)
  • 特點:
  • 在成員函數中可通過this指針差別成員變量與形參變量
  • this可以顯式調用
  • 示例代碼:
class Cperson
{
private:
int age;
float height;
public:
void InitPerson(int age,float height);
};
void Cperson::InitPerson(int age,float height)
{
this->age=age;
this->height=height;
}      

二、傳回*this成員函數

  • 概念:傳回值是*this,也就是傳回調用此成員函數的對象的自身引用,傳回值類型為對象引用類型
class Person
{
private:
int age;
public:
Person& setAge(int age);//傳回自身引用
};
Person& Person::setAge(int age)
{
this->age=age;
return *this;//傳回*this(自身引用)
}      
  • 從const成員函數傳回*this:如果一個const成員函數傳回*this,那麼此函數的傳回值類型是常量引用:比如上面的代碼中的setAge函數變為常量成員函數
const Person& setAge(int age)const;      

三、this指針剖析

this指針剖析

  • 對于類成員函數而言,并不是一個對象對應一個單獨的成員函數體,而是此類的所有對象共用這個成員函數體。當程式被編譯之後,此成員函數位址已經确定
  • 我們常說,調用類的成員函數時,會将目前對象的this指針傳遞給成員函數:
  • 上面我們說了,一個類的成員函數隻定義了一份,所有對象共用這個成員函數體
  • 那麼如何區分哪個對象調用此函數呢?這就是this指針的作用,每次對象調用成員函數時,将this指針(對象首位址)傳遞給成員函數,然後在函數體内對類的資料成員通路時,就會被轉換為“this->資料成員”的形式
  • 如果成員函數内部沒有通路對象的任何資料成員,那麼傳進來的this指針實際沒有任何用處,這樣的函數與全局函數并沒有太大差別

示範案例

#include <iostream>
using namespace std;
class MyClass
{
public:
int data;
MyClass(int data)
{
this->data = data;
}
void print()
{
//cout<< data << endl;
cout << "hello!" << endl;
}
};
int main()
{
MyClass *pMyClass;
pMyClass = new MyClass(1);
pMyClass->print();
pMyClass[0].print();
pMyClass[1].print();
pMyClass[10000000].print();
return 0;
}      
  • 運作效果如下所示:
  • 結果分析: 
  • 此處我們建立了一個對象,但是通路後面越界之後仍然可以通路成功
  • 上面我們說過了,如果一個成員函數沒有操作任何類的資料成員,是以這個成員函數與全局函數類型,調用之後都會列印“hello!”
  • 但是如果把14行的代碼取消注釋會怎麼樣呢?見下面示範案例

代碼改進

#include <iostream>
using namespace std;
class MyClass
{
public:
int data;
MyClass(int data)
{
this->data = data;
}
void print()
{
//cout<< data << endl;
cout << "hello!" << endl;
}
};
int main()
{
MyClass *pMyClass;
pMyClass = new MyClass(1);
pMyClass->print();
pMyClass[0].print();
pMyClass[1].print();
pMyClass[10000000].print();
return 0;
}      
  •  運作效果如下所示:
C++:19---this指針
  • 結果分析: 
  • 此處我們的成員函數通路了類的資料成員,那麼每次列印資料時都會通過this指針調用
  •  但是此處我們隻建立了一個對象,是以當調用pMyClass[1]和pMyClass[10000000]越界了,找不到對象,進而報錯