请看下边代码
#include <stdio.h>
class A
{
public:
void print()
{
printf("123456\n");
};
int a;
};
int main()
{
A *p = nullptr;
p->print();//会不会崩溃
((A*)0xAA)->print();
return 0;
}
上边代码看起来很简单,其实内部引擎c++类成员方法的调用逻辑,很多人第一眼反应是会崩溃,答案恰恰相反,能够正常打印出123456,接下来就分析下原因,
c++类方法在编译编译后,printf方法会变成printf(A *a); 所以上边printf展开后就变成下边
void A::print(A *a)
{
printf("123456\n");
};
也就是说不论你a是空指针,还是任意数,方法里边压根没有访问,所以会正常打印。理解这个我们在看下下边代码会怎么样?
#include <stdio.h>
class A
{
public:
virtual void print()
{
printf("123456\n");
};
int a;
};
int main()
{
A *p = nullptr;
p->print();//会不会崩溃
return 0;
}
相比一开始的时候,就加了virtual 关键字,这里肯定会崩溃的。为什么呢,原因在与加了virtual printf就是一个虚方法,在调用它之前需要拿到a的虚表指针,然后在查表找到printf的调用地址,最后在调用printf, 看的出来p指向的是空指针,空指针访问虚表指针,非法访问,就会崩溃了。