多态的定义、重写、原理

多态是面向对象编程中的一个重要概念,它允许通过父类引用调用子类的重写方法。文章详细介绍了多态的定义、实现条件,包括虚函数的作用、函数隐藏与重写、析构函数的重写以及final和override关键字。此外,还讨论了抽象类、虚函数表以及单继承和多继承下的虚函数表结构。
摘要由CSDN通过智能技术生成

多态

多态image-20230311114543868

多态的定义和条件

定义:多态是在不同继承关系的对象上,去调用同一函数,从而产生不同的行为。

在继承中构成多态还需要两个条件:

一是被调用的函数必须是虚函数(函数用virtual关键字修饰)。并且要求父类和子类的虚函数符合三同即函数名、参数、返回值类型相同,即为虚函数的重写/覆盖(子类的虚函数重写了父类虚函数)

二是必须是父类的指针、引用去调用虚函数

如图,有个Person类里面实现了一个BuyTicket函数,还有个student类里面也实现了一个BuyTicket函数。此时student类继承了Person类,两者的BuyTicket函数都是虚函数满足函数名、参数、返回值相同,并且在fun函数里参数是用的父类的引用去调用。此时Person和student的BuyTicket就构成了多态。通过不同的对象去调用同一个函数产生了不同的行为!

image-20230310093427991

另外,子类的函数可以不是虚函数,但父类的函数必须是虚函数

image-20230310111016688

协变(父类和子类的返回值类型不同)

三同中,返回值类型可以不同,但要求返回值是父子类关系的一个引用或者指针

image-20230310111956645

image-20230310112309224

就算用的别的类型也可以。这里我创建了一个父子类关系类型A,类型B用来做返回值

image-20230310112716213

函数隐藏和虚函数重写的比较

我们知道,父类和子类的函数名相同就构成了函数隐藏或者重定义。而多态的要求比隐藏更严格,虚函数的重写必须满足三同(函数名、参数、返回值类型相同),其中一个不相同即为函数隐藏。

函数重载函数隐藏/重定义函数重写/覆盖
同一作用域;函数名相同;参数列表不同(参数类型、个数、顺序);返回值不影响不同作用域(父类和子类);函数名相同;参数列表不同时,基类有无virtual修饰都是;参数列表相同时,基类没有virtual修饰是;返回值可以不同不同作用域(父类和子类);函数名相同;参数列表相同;返回值类型相同;基类必须要有virtual修饰;必须是由父类的引用或者指针调用虚函数;返回值类型不同时,返回值类型也必须是父子类关系的指针或者引用—协变

由此可见,多态调用与调用的对象有关,普通调用与调用的对象类型有关

析构函数的重写

这里是普通调用析构函数,目前没什么问题

image-20230310120153798

然而当有父类的指针指向或者父类的引用时,子类的析构函数没有执行,产生了内存泄漏。原因:子类的切片,指针或者引用指向父类那部分,所以子类就只调用了父类的析构函数。

image-20230310120400315

这时候就需要用到函数的重写。

只需要给父类的析构函数加上virtual修饰即可。编译后,编译器对父类和子类的析构函数名称都统一处理成destructor

image-20230310121153535

关键字final和override

前面都介绍的是如何实现函数的重写,那么一个虚函数不想被重写呢?

给虚函数加上关键词final加以修饰表示虚函数不能被重写

image-20230310163252125

那一个类不想被继承呢?

一是构造函数私有

image-20230310163942372

二是用final修饰,即可理解为最后的类

image-20230310164121328

override修饰子类函数可用来在编译期间检查子类函数是否对父类函数完成了重写

image-20230310164459296

抽象类

定义:在虚函数后面写=0,则这个函数为纯虚函数。包括纯虚函数的类称抽象类或接口类。抽象类不能实例化出对象,其子类也不能实例化出对象,除非子类重写了纯虚函数。纯虚函数规定了子类必须重写,即接口继承。

image-20230310171946833

image-20230310172204782

虚函数继承通过与普通函数的继承对比,普通函数继承为实现继承,派生类继承了基类,可以用基类的函数。而虚函数虚函数是一种接口继承。派生类继承的是接口,目的是为了重写,达到多态。

多态的原理

接下来看一个含有虚函数的类的大小

类A里有一个int类型和一个char类型,合计5个字节,加上虚函数dave,虚函数里有虚表指针4个字节(32位系统下),合计9个字节,内存对齐后是12个字节

image-20230310183256489

我们打开调试窗口可以看到有个指针_vfptr

image-20230310183757565

那如果类里多几个虚函数呢??

class A
{
public:
	virtual void dave1(){}
	virtual void dave2(){}
	virtual void dave3(){}
private:
	
	int _a;
	char _b;
};

int main()
{
	A aa;
	cout << "带有虚函数的类的大小:" << sizeof(aa) << endl;

	return 0;
}

类里再多的虚函数也只有一个虚表指针,指针指向一个虚函数表,表里存放着指向各个虚函数的指针,该虚表本质上是函数指针数组。

image-20230310185721544

class A
{
public:
	virtual void Func1()
	{
		cout << "A::Func1()" << endl;
	}
	virtual void Func2()
	{
		cout << "A::Func2()" << endl;
	}
	void Func3()
	{
		cout << "A::Func3()" << endl;
	}
private:
	int _a = 1;
};
class B : public A
{
public:
	virtual void Func1()
	{
		cout << "B::Func1()" << endl;
	}
private:
	int _b = 2;
};
int main()
{
	A a;
	B b;
	return 0;
}

这里B类继承了A类,并完成了对虚函数fun1的重写,而没有对A类的虚函数fun2重写。可以看到两个虚函数都继承了下来,但fun1的地址该变了,而fun2的地址没有改变。可以猜测:子类在对父类函数的重写时,是先把父类的虚函数表拷贝一份,然后对要重写的函数进行覆盖。

image-20230310191955572

那普通调用和多态调用的原理有差别吗?

这里ptr指针对fun3函数调用为普通调用,而对fun1函数调用为多态调用

image-20230310193845725

调试时转到反汇编,可以看到普通调用是直接call函数,而多态调用则步骤很多,还用到了各种寄存器。

这里更加应证了普通调用为编译时绑定,即在编译期间就确定了程序的行为,也称静态多态,比如函数重载。

而多态调用为运行时绑定,在程序运行期间根据具体的类型确定程序的行为,调用具体的函数,也称动态多态。

image-20230310194240289

实际上,普通调用时,是根据指针指向的类型进行调用。ptr指向b对象的fun3是A类fun3的切片,跟ptr指向a对象的fun3无异。所以是直接call A类的fun3函数。

而多态调用是根据指针指向对象的类型有关。ptr指向b对象的fun1,**由于fun1是虚函数,该指向虚函数的指针进入了虚数表,那么指针就进入虚数表里找,找到的是类型B对类型A重写的fun1虚函数的指针,那么调用的就是重写的fun1函数,注意该切片部分是被重写的!**而ptr指向a对象的fun1也是进入虚数表里找,找到的调用的即是fun1虚函数本身。

而多态能完成指向谁调用谁其根本就是由于虚数表。

那虚表在哪里呢?

找到虚表存放的第一个指针的地址就能找到虚表的位置。

int main()
{
	int a = 0;
	cout << "栈:" << &a << endl;

	int* p1 = new int;
	cout << "堆:" << p1 << endl;

	const char* str = "hello world";
	cout << "代码段/常量区:" << (void*)str << endl;

	static int b = 0;
	cout << "静态区/数据段:" << &b << endl;

	A aa;
	cout << "虚表:" << (void*)*((int*)&aa) << endl;
	return 0;
	}

image-20230310201104960

通过测试,可以看到虚表的位置离代码段和静态区很近

并且同个类型的虚表是共享的。

image-20230310201806013

单继承和多继承的虚函数表

单继承下的虚函数表

接下来来看派生类对象的虚数表模型

typedef void(*vfptr)();//定义了函数指针
void PrintVtalbe(vfptr vtable[])//传函数指针数组
{
    for (int i = 0; vtable[i] != nullptr; i++)
    {
        printf("[%d]:%p->", i, vtable[i]);
        vtable[i]();    
    }
    cout << endl;
}
class A {
public:
    virtual void func1() { cout << "Base::func1" << endl; }
    virtual void func2() { cout << "Base::func2" << endl; }
private:
    int _a;
};
class B :public A {
public:
    virtual void func1() { cout << "Derive::func1" << endl; }
    virtual void func3() { cout << "Derive::func3" << endl; }
private:
    int _b;
};
int main()
{
    A a;
    B b;
    vfptr* vtab1 = (vfptr*)(*((void**)&a));
    vfptr* vtab2 = (vfptr*)(*((void**)&b));
    PrintVtalbe(vtab1);
    PrintVtalbe(vtab2);
    return 0;
}

通过调试窗口,可以看到b对象只有继承下来的fun1和fun2而没有fun3

image-20230311095526878

通过调用内存窗口可以看到,b对象的第二个地址和a对象的第二个地址相同,推测那个就是fun2,而a对象的第三个地址就是空,b对象的第四个地址才是空,可以推测虚数表是以空结尾。那么b对象的第三个地址就是fun3,fun3进虚数表但是不在调试窗口上显示!

image-20230311095816467

通过打印就可以得到虚数表地址了

image-20230311100052995

多继承下的虚函数表

多继承派生类的未重写的虚函数放在第一个继承基类部分的虚函数表中

另外,inline函数可以是虚函数吗?

inline在调用的展开,也就没有了地址,inline函数没有地址放到虚数表里。但多态调用inline函数是可以编译通过的,但忽略了inline的特性;而普通调用仍保持inline特性。

静态成员可以是虚函数吗?

不可以!静态成员没有this指针,且静态成员本身就不能实现多态。

构造函数可以是虚函数吗?
虚数表指针是在初始化列表时初始化,构造函数若是虚函数则虚数表无法初始化。

对象访问普通函数和虚函数谁更快?

如果是普通调用,则一样快。但如果是多态调用,则普通函数更快。运行时调用虚函数需要到虚数表里面去查找函数地址。

虚函数表在编译阶段生成,但虚函数表指针在运行时构造函数列表初始化。

  • 18
    点赞
  • 15
    收藏
    觉得还不错? 一键收藏
  • 28
    评论
在Java中,多态性是面向对象编程的重要特性之一。它允许一个对象变量引用不同类型的对象,并在运行时调用相应对象的方法。多态性的实现原理主要依赖于以下两个概念: 1. 继承(Inheritance):Java中的继承机制允许一个类继承另一个类的属性和方法。子类可以继承父类的方法并重写它们。 2. 方法重写(Method Overriding):子类有权重写从父类继承的方法,即在子类中重新定义与父类相同名称、参数列表和返回类型的方法。 当使用多态性时,实际上是通过父类类型的引用变量来引用不同子类对象。具体实现过程如下: 1. 定义一个父类(基类)并声明一个方法。 2. 在子类中继承这个父类并重写父类中的方法。 3. 创建一个父类引用变量,并将其指向一个子类对象。 4. 通过父类引用变量调用方法。在运行时,会根据具体对象的类型来调用相应的方法。 例如,假设我们有一个Animal类作为父类,有Cat和Dog作为子类: ```java class Animal { public void makeSound() { System.out.println("Animal is making a sound"); } } class Cat extends Animal { @Override public void makeSound() { System.out.println("Meow"); } } class Dog extends Animal { @Override public void makeSound() { System.out.println("Woof"); } } public class Main { public static void main(String[] args) { Animal animal1 = new Cat(); Animal animal2 = new Dog(); animal1.makeSound(); // 调用Cat类的makeSound方法,输出 "Meow" animal2.makeSound(); // 调用Dog类的makeSound方法,输出 "Woof" } } ``` 在上面的例子中,animal1和animal2都是Animal类的引用变量,但它们分别指向Cat和Dog对象。通过调用相同的方法makeSound,实际上调用了不同类中的对应重写方法,实现了多态性。
评论 28
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值