多态的定义、重写、原理
创始人
2024-06-01 17:41:39
0

多态

多态image-20230311114543868

文章目录

  • 多态
    • 多态的定义和条件
      • 协变(父类和子类的返回值类型不同)
        • 函数隐藏和虚函数重写的比较
      • 析构函数的重写
      • 关键字final和override
    • 抽象类
    • 多态的原理
      • 单继承和多继承的虚函数表
        • 单继承下的虚函数表
        • 多继承下的虚函数表

多态的定义和条件

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

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

一是被调用的函数必须是虚函数(函数用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指针,且静态成员本身就不能实现多态。

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

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

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

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

相关内容

热门资讯

Python|位运算|数组|动... 目录 1、只出现一次的数字(位运算,数组) 示例 选项代...
张岱的人物生平 张岱的人物生平张岱(414年-484年),字景山,吴郡吴县(今江苏苏州)人。南朝齐大臣。祖父张敞,东...
西游西后传演员女人物 西游西后传演员女人物西游西后传演员女人物 孙悟空 六小龄童 唐僧 徐少华 ...
名人故事中贾岛作诗内容简介 名人故事中贾岛作诗内容简介有一次,贾岛骑驴闯了官道.他正琢磨着一句诗,名叫《题李凝幽居》全诗如下:闲...
和男朋友一起优秀的文案? 和男朋友一起优秀的文案?1.希望是惟一所有的人都共同享有的好处;一无所有的人,仍拥有希望。2.生活,...
戴玉手镯的好处 戴玉手镯好还是... 戴玉手镯的好处 戴玉手镯好还是碧玺好 女人戴玉?戴玉好还是碧玺好点佩戴手镯,以和田玉手镯为佳!相嫌滑...
依然什么意思? 依然什么意思?依然(汉语词语)依然,汉语词汇。拼音:yī    rán基本解释:副词,指照往常、依旧...
高尔基的散文诗 高尔基的散文诗《海燕》、《大学》、《母亲》、《童年》这些都是比较出名的一些代表作。
心在飞扬作者简介 心在飞扬作者简介心在飞扬作者简介如下。根据相关公开资料查询,心在飞扬是一位优秀的小说作者,他的小说作...
卡什坦卡的故事赏析? 卡什坦卡的故事赏析?讲了一只小狗的故事, 我也是近来才读到这篇小说. 作家对动物的拟人描写真是惟妙...
林绍涛为简艾拿绿豆糕是哪一集 林绍涛为简艾拿绿豆糕是哪一集第三十二集。 贾宽认为是阎帅间接导致刘映霞住了院,第二天上班,他按捺不...
小爱同学是女生吗小安同学什么意... 小爱同学是女生吗小安同学什么意思 小爱同学,小安同学说你是女生。小安是男的。
内分泌失调导致脸上长斑,怎么调... 内分泌失调导致脸上长斑,怎么调理内分泌失调导致脸上长斑,怎么调理先调理内分泌,去看中医吧,另外用好的...
《魔幻仙境》刺客,骑士人物属性... 《魔幻仙境》刺客,骑士人物属性加点魔幻仙境骑士2功1体质
很喜欢她,该怎么办? 很喜欢她,该怎么办?太冷静了!! 太理智了!爱情是需要冲劲的~不要考虑着考虑那~否则缘...
言情小说作家 言情小说作家我比较喜欢匪我思存的,很虐,很悲,还有梅子黄时雨,笙离,叶萱,还有安宁的《温暖的玄》 小...
两个以名人的名字命名的风景名胜... 两个以名人的名字命名的风景名胜?快太白楼,李白。尚志公园,赵尚志。
幼儿教育的代表人物及其著作 幼儿教育的代表人物及其著作卡尔威特的《卡尔威特的教育》,小卡尔威特,他儿子成了天才后写的《小卡尔威特...
海贼王中为什么说路飞打凯多靠霸... 海贼王中为什么说路飞打凯多靠霸气升级?凯多是靠霸气升级吗?因为之前刚到时确实打不过人家因为路飞的实力...
运气不好拜财神有用吗运气不好拜... 运气不好拜财神有用吗运气不好拜财神有没有用1、运气不好拜财神有用。2、拜财神上香前先点蜡烛,照亮人神...