虚函数 & 多态

 

  • 虚函数在类的成员函数前加virtual关键,这时成员函数就是虚函数。这里需要注意的是必须是类的成员函数。

  • 虚函数表通过一块连续的内存来存储虚函数地址。在对象的模型里存着。如下所示:

 

class Base
{
public:
	
	virtual void Fun1()
	{
		cout << "Base::Fun1()" << endl;
	}

	virtual void Fun2()
	{
		cout << "Base::Fun2()" << endl;
	}

public:
	int _Base;
};

class Derive:public Base
{
public:
	virtual void Fun1()
	{
		cout << "Derive::Fun()" << endl;
	}

	virtual void Fun3()
	{
		cout << "Derive::Fun3()" << endl;
	}

	virtual void Fun4()
	{
		cout << "Derive::Fun4()" << endl;
	}
public:
	int _Derive;

};

上述的图示(单继承):

 

 

菱形虚拟继承:(多继承)

下面通过一段简单的菱形继承的代码来分析:

//菱形虚拟继承博客演示

class A
{
public:
	virtual void f1()
	{
		cout << "A::f1()" << endl;
	}

	virtual void f2()
	{
		cout << "A::f2()" << endl;
	}

	int _a;
};

class B :virtual public A
{
public:
	virtual void f1()
	{
		cout << "B::f1()" << endl;
	}
	
	virtual void f3()
	{
		cout << "B::f3()" << endl;
	}

	int _b;
};

class C :virtual public A
{
public:
	virtual void f1()
	{
		cout << "C::f1()" << endl;
	}

	virtual void f4()
	{
		cout << "C::f4()" << endl;
	}

	int _c;
};

class D :public B, public C
{
public:

	virtual void f1()
	{
		cout << "D::f1()" << endl;
	}

	virtual void f5()
	{
		cout << "D::f5()" << endl;
	}

	int _d;
};

typedef void(*VfTables)() ;               //函数指针
void printVfptr(int **VfTable)
{
	printf("虚表的地址:%p \n",VfTable);
	for (size_t i = 0; VfTable[i] != 0; ++i)
	{
		printf("VfTable[%d] :%p -> ",i,VfTable[i]);
		VfTables func = (VfTables)VfTable[i];
		func();
	}
}

int main()
{

	D d;
	d._a = 1;
	d._b = 2;
	d._c = 3;
	d._d = 4;

	cout << "====================================" << endl;
	printVfptr(((int**)(*(int**)&d)));
	cout << "====================================" << endl;
	printVfptr((int**)(*((int**)((char*)&d + sizeof(B)-sizeof(A)))));
	cout << "====================================" << endl;
	printVfptr((int**)(*(int**)((char*)&d + sizeof(D)-sizeof(A))));
	cout << "====================================" << endl;

	return 0;
}

图示分析:

 

  • 多态即多种形态

 

触发多态的条件:1.父类的指针或引用调用虚函数

                              2.虚函数的重写(覆盖)

 

  • 为什么多态非要是对象的指针或者引用呢?
  • 否则就是值拷贝,不拷贝虚表

下面代码实现多态

class A
{
public:
	virtual void Fun1()
	{
		cout << "A::Fun1" << endl;
	}

	void GetFun()
	{
		Fun1();
	}
};

class B :public A
{
public:
	virtual void Fun1()
	{
		cout << "B::Fun1" << endl;
	}
};

int main()
{
	A a;
	B b;
	A* pa = &a;
	pa->GetFun();

	pa = &b;
	pa->GetFun();
}

这里就可以看出来的,pa是一个A*的类型,但是通过同一类型的指针,调了两个不同的函数,一个是父类的Fun1,一个是子类的Fun1。

 

重写,重载,隐藏的区别:

 

 

 

 

 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
虚函数是实现C++多态性的重要手段。通过定义虚函数,可以在基类和派生类之间建立动态绑定关系,实现多态。 在C++中,定义虚函数需要在函数声明前面加上`virtual`关键字,例如: ``` class Base { public: virtual void func(); }; class Derived : public Base { public: void func(); }; ``` 在上面的代码中,`Base`类中定义了一个虚函数`func`,派生类`Derived`中也定义了一个同名的函数`func`。由于`Base`类中的`func`函数被声明为虚函数,因此在基类指针指向派生类对象时,调用`func`函数时会根据对象实际的类型来进行动态绑定,从而调用派生类中的`func`函数。 例如: ``` Base *ptr = new Derived(); ptr->func(); // 调用Derived类中的func函数 ``` 上面的代码中,`ptr`是一个指向`Derived`类对象的基类指针。调用`func`函数时,会根据`Derived`类的实际类型进行动态绑定,从而调用`Derived`类中的`func`函数。 虚函数的实现机制是通过为每个对象添加一个虚函数表(VTable)来实现的。虚函数表是一个指向虚函数地址的指针数组,每个对象都有一个指向其对应的虚函数表的指针。在调用虚函数时,首先会根据对象的虚函数表找到对应的函数地址,然后进行调用。 总的来说,虚函数实现了C++中的动态绑定机制,是多态性的关键。通过定义虚函数,可以在基类和派生类之间建立动态绑定关系,实现多态

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值