類中虛函數、普通成員函數的調用

class A
{
public:
	virtual void P1() {
		printf("A1\n");
	}
	void P2() {
		printf("A2\n");
	}
};
class B : public A
{
public:
	virtual void P1() {
		printf("B1\n");
	}
	void P2() {
		printf("B2\n");
	}
};

int main(int argc, char **argv)
{
	A *pA1 = new A();
	B *pB1 = (B *)pA1;
	pB1->P1();//A1
	pB1->P2();//B2

	printf("----------------\n");

	B *pB2 = new B();
	A *pA2 = (A *)pB2;
	pA2->P1();//B1 :多態,類B中虛成員函數表中,保存的是B中函數地址
	pA2->P2();//A2 :強制轉換,按照A類成員訪問方式使用B實例,調用A成員函數

	printf("----------------\n");

	A *pA3 = NULL;
	B *pB3 = (B *)pA3;
	pB3->P1();//crash : 在調用虛函數時,會先尋找this,再根據this指向的類實例找到虛函數表指針,再偏移找到調用函數。由於this爲NULL,crash
	pB3->P2();//B2 :直接找到函數地址,並調用,過程中未使用到this,可正常調用,根據類類型訪問到的是B中成員函數

	printf("----------------\n");

	B *pB4 = NULL;
	A *pA4 = pB4;
	pA4->P1();//crash
	pA4->P2();//A2
	pB4->P2();//B2

	printf("----------------\n");

	system("pause");
	return 0;
}

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章