C++多态&多态模型

                     多态&多态模型

一、什么是多态?

  多态顾名思义就是一个事物有多种形态,在C++程序设计中,多态性是指具有不同功能的函数可以用同一个函数名,这样就可以用一个函数名调用不同内容的函数,即“一个接口,多种方法”。

多态分为两种:

  (1)静态多态(重载函数、模板等);(2)动态多态(虚函数等)。

构成多态的条件:

(1)虚函数的重写。(子类可以不用写virtual,但父类必须写virtual,一般最好子类和父类都加virtual)
(2)父类的指针(或引用)调用虚函数。 

  虚函数的作用主要就是实现多态,简单的说父类的指针或引用调用重写的虚函数的时候,当父类指针或者引用指向父类对象时调用的是父类的虚函数,指向子类对象时调用的是子类的虚函数。 

二、  多态的对象模型--单继承&多继承

1.单继承

继承规则: 
(1)如果派生类没有重写基类的虚函数,并且没有添加自己的虚函数,则直接继承基类的虚函数; 
(2)若派生类新增了虚函数,则虚表先存放基类继承来的虚函数,再存放派生类的虚函数,之后存放派生类自己的虚函数。 
(3)若基类中有虚函数被派生类重写,派生类中重写的虚函数会替换基类中被重写虚函数的位置,之后存放派生类新添加的虚函数。


代码说明:

#include<iostream>
using namespace std;

class A
{
public:
	virtual void func1()
	{
		cout << "A::func1" << endl;
	}
	virtual void func2()
	{
		cout << "A::func2" << endl;
	}
protected:
	int _a;
};
class B :public A
{
public:
	virtual void func1()
	{
		cout << "B::func1" << endl;
	}
	virtual void func3()
	{
		cout << "B::func3" << endl;
	}
protected:
	int _b;
};
int main()
{
	A a;
	B b;
	system("pause");
	return 0;
}


虚函数B::func3()没有放进去,其实监视窗口是会骗人的,我们想要弄清楚它们是如何存放的,那么就需要自己打印虚函数表。

typedef void(*V_FUNC)();
void PrintVTable(int* Vtable)
{
	printf("vtable:0x%p\n", Vtable);
	int** ppVtable = (int**)Vtable;
	for (size_t i = 0; ppVtable[i] != 0; ++i)
	{
		printf("Vtable[%u]:0x%p->", i , ppVtable[i]);
		V_FUNC f = (V_FUNC)ppVtable[i];
		f();
	}
	cout << "*********************************" << endl;
}
int main()
{
	A a;
	B b;
	PrintVTable(*((int**)&a));
	PrintVTable(*((int**)&b));
	system("pause");
	return 0;
}
结果显示:


画出单继承的对象模型:


2.多继承

继承规则:没有重写的多函数的虚表,先按照继承次序打印第一个基类虚表,在打印派生类新添加的虚表,之后打印第二个基类的虚表。若有重写则转变为单继承问题。

                         

代码实现:同理,从单继承可得,我们同样需要手动打印观察虚函数的存放

#include<iostream>
using namespace std;

class A
{
public:
	virtual void func1()
	{
		cout << "A::func1" << endl;
	}
	virtual void func2()
	{
		cout << "A::func2" << endl;
	}
	int _a;
};
class B 
{
public:
	virtual void func1()
	{
		cout << "B::func1" << endl;
	}
	virtual void func3()
	{
		cout << "B::func3" << endl;
	}
	int _b;
};
class C:public A,public B
{
public:
	virtual void func1()
	{
		cout << "C::func1" << endl;
	}
	virtual void func4()
	{
		cout << "C::func4" << endl;
	}
	int _c;
};
typedef void(*V_FUNC)();
void PrintVTable(int* Vtable)
{
	printf("vtable:0x%p\n", Vtable);
	int** ppVtable = (int**)Vtable;
	for (size_t i = 0; ppVtable[i] != 0; ++i)
	{
		printf("Vtable[%u]:0x%p->", i , ppVtable[i]);
		V_FUNC f = (V_FUNC)ppVtable[i];
		f();
	}
	cout << "*********************************" << endl;
}
int main()
{
	A a;
	B b;
	C c;
	c.A::_a = 1;
	c.B::_b = 2;
	c._a = 3;
	c._b = 4;
	c._c = 5;
	PrintVTable(*((int**)&c));
	PrintVTable(*((int**)((char*)&c+sizeof(A))));
	system("pause");
	return 0;
}
多继承的动态模型:







 

 

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值