【C++】类和对象2:this指针

前言

今天来学习this指针

引入

我们用一个日期Date类来举个例子

class Date
{ 
public:
	void Init(int year, int month, int day)
 	{
 		_year = year;
	 	_month = month;
 		_day = day;
 	}
 	void Print()
	{
 		cout <<_year<< "-" <<_month << "-"<< _day <<endl;
 	}
private:
	int _year;
	int _month;
	int _day;
};

int main()
{
	Date d1,d2;
	d1.Init(2024,1,3);
	d2.Init(2024,2,3);
	d1.Print();
	d2.Print();
	return 0}

看完上面这段代码 有个疑问:
为什么调用的是同一个函数(不理解为什么是同一个函数可以看上一篇文章) 但打印出来一个是

2024 1 3

一个却是

2024 2 3

解释

上篇文章提到过 成员的定义是根据对象来实现的
在类的定义中 只是成员的声明 并未真实创建

那么 所有的成员函数都会多出一个指针 名为this指针 是成员函数的第一个参数

C++编译器给每个“非静态的成员函数“增加了一个隐藏的指针参数,让该指针指向当前对象(函数运行时调用该函数的对象),在函数体中所有“成员变量”的操作,都是通过该指针去访问。只不过所有的操作对用户是透明的,即用户不需要来传递,编译器自动完成。

所以Date类中的成员函数和函数调用其实可以理解为:

 	void Print(Date* const this)
	{
 		cout <<this->_year<< "-" <<this->_month << "-"<< this->_day <<endl;
 	}

	d1.Print(&d1);

其实和C没啥区别 只是创建者自己优化了一下 把传参这一步交给了编译器

this指针的特性

  1. this指针的类型:类类型*const,即成员函数中,不能给this指针赋值。
  2. 只能在“成员函数”的内部使用
  3. this指针本质上是“成员函数”的形参,当对象调用成员函数时,将对象地址作为实参传递给this形参。所以对象中不存储this指针。
  4. this指针是“成员函数”第一个隐含的指针形参,一般情况由编译器通过ecx寄存器自动传递,不需要用户传递

面试题

  1. this指针存在哪里?
    this指针是形参 存在栈里面
    (部分编译器用寄存器存储,如VS)
  2. this指针可以为空吗?
// 1.下面程序编译运行结果是? A、编译报错 B、运行崩溃 C、正常运行
class A
{ 
public:
    void PrintA() 
   {
        cout<<_a<<endl;
   }
private:
 int _a;
};
int main()
{
    A* p = nullptr;
    p->PrintA();
    return 0; 
}

p传递了一个this指针(空指针) 并且在输出_a的时候对空指针进行了解引用
所以是运行崩溃

那么留一道思考题:

// 1.下面程序编译运行结果是? A、编译报错 B、运行崩溃 C、正常运行
class A
{ 
public:
    void PrintA() 
   {
        cout<<this<<endl;
   }
private:
 int _a;
};
int main()
{
    A* p = nullptr;
    (*p).PrintA();
    return 0; 
}

结语

关于this指针的学习到这里就结束了 我们下篇文章见~

你可能感兴趣的:(C++知识点,c++,java,jvm)