如何计算类对象的大小
问题:类中既可以有成员变量,又可以有成员函数,那么一个类的对象中包含了什么?如何计算一个类的大小?
类对象的存储方式
只保存成员变量,成员函数存放在公共的代码段
结论:一个类的大小,实际就是该类中”成员变量”之和,当然要注意内存对齐
注意空类的大小,空类比较特殊,编译器给了空类一个字节来唯一标识这个类的对象。
不然分配0字节无法区分实例化的对象,也无法取地址
this指针
问题:既然成员函数存在公共区里,那么调用的都是同一个Print函数,那为什么d1.Print()和d2.Print()调用结果不一样呢?
原因:C++引入了this指针,在传参数时,隐含的把对象d1,d2的地址传入成员函数里。
编译器隐含的帮我们做了如下的事情
注意:this不能在形参和实参显示传递,但是可以在函数内部显示使用
- this指针存在哪里?
this是成员函数的形参,所以this指针是跟普通参数一样存在函数调用的栈帧里面。
因为需要频繁使用this指针,vS下面对this指针传递,进行优化,对象地址是放在ecx,ecx存储this指针的值
- this指针可以为空吗?
样例1
不解引用的话,向this传递一个空指针是可以的
// 1.下面程序编译运行结果是? A、编译报错 B、运行崩溃 C、正常运行
class A
{
public:
void Print()
{
cout << "Print()" << endl;
}
private:
int _a;
};
int main()
{
A* p = nullptr;
p->Print();
return 0;
}
p->Print();等价于(*p).Print()
p调用Print()
那么它就真的解引用了吗,不是,Print()的地址在公共区,并不是在对象p里面,那解引用p是干嘛呢?。
底层这样的,p会作为实参传递给this指针。(和取对象的地址一样,这里直接是指针,所以直接mov给ecx)
只不过this指针是nullptr
this指针是空的,但是函数内没有对this指针解引用
所以正常运行
样例2
传nullptr后对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;
}
以上两个问题排除编译报错,因为空指针的问题是运行报错
总结
对比C和C++
this指针隐含的帮助我们完成了对象的取地址操作,更为方便