如果一个派生类有多个直接基类,而这些直接基类又有一个共同的基类,则在最终的派生类中会保留该间接共同基类数据成员的多份同名成员。C++提供虚基类的方法,使得在继承间接共同基类时只保留一份成员。...现在,将类A声明为虚基类,方法如下: class A//声明基类A {…}; class B :virtual public A//声明类B是类A的公用派生类,A是B的虚基类 {…}; class...C :virtual public A//声明类C是类A的公用派生类,A是C的虚基类 {…}; 注意: 虚基类并不是在声明基类时声明的,而是在声明派生类时,指定继承方式时声明的。...因为一个基类可以在生成一个派生类时作为虚基类,而在生成另一个派生类时不作为虚基类。...下面我们把A改成虚基类再看看结果。
---- 摘自《C++程序设计》 如果一个派生类有多个直接基类,而这些直接基类又有一个共同的基类,则在最终的派生类中会保留该间接共同基类数据成员的多份同名成员。...C++提供虚基类(virtual base class)的方法,使得在继承间接共同基类时只保留一份成员。 ---- 下面举例说明: 在如下的图中: ?...如果使用虚基类的话,Graduate将有两份age拷贝,两份gender拷贝,两份name拷贝,一个来自Student,一个来自Teacher。...所以C++中提出了虚基类的实现方式。...声明虚基类的一般形式是: class 派生类名:virtual 继承方式 基类名称 下面是上面实例的代码: 类声明person.h: #pragma once #include
1.多重继承带来的问题 C++虚拟继承一般发生在多重继承的情况下。C++允许一个类有多个父类,这样就形成多重继承。...(1)当在多条继承路径上有一个公共的基类,在这些路径中的某几条汇合处,这个公共的基类就会产生多个实例(或多个副本),若只想保存这个基类的一个实例,可以将这个公共基类说明为虚基类,就像Student和Employee...(2)被虚拟继承的基类,叫做虚基类。虚基类实际指的是继承的方式,而非一个基类,是动词,而非名词。 (3)为了实现虚拟继承,派生类对象的大小会增加4。...这个增加的4个字节,是因为当虚拟继承时,无论是单虚继承还是多虚继承,派生类需要有一个虚基类表来记录虚继承关系,所以此时子类需要多一个虚基类表指针,而且只需要一个即可。...(4)虚拟继承中,虚基类对象是由最远派生类的构造函数通过调用虚基类的构造函数进行初始化的,派生类的构造函数的成员初始化列表中必须列出对虚基类构造函数的调用;如果未列出,则表示使用该虚基类的缺省构造函数。
参考链接: C++继承 继承 类和类的关系有组合、继承和代理。继承的本质就是代码复用。子类继承父类中的一些东西,父类也称为基类,子类也称为派生类。派生类继承了基类除构造函数以外的所有成员。 ...的布局如下: 由图可见,基类的布局优先级高于派生类。...1.调用基类的构造函数 2.调用派生类的构造函数 派生类的析构可想而知: 1.调用派生类的析构函数 2.调用基类的析构函数 虚函数 如下程序: class Base { public: Base...基类中含有虚函数,那么基类布局中存在一个虚函数指针,指向虚函数表;且其派生类中与其同名同参的函数不需要加virtual也是虚函数。...此时基类和派生类的布局如下: vfptr的指针大小为4(32位机器)。因此基类字节数为8,派生类为12。
虚基类并不是“绝对的”,而是“相对的”:虚基类在它自身声明、定义的时候无需任何修饰,只是在子类继承时进行 virtual 修饰。...同时,在虚继承机制当中,虚基类是由最终的派生类进行初始化的,本身达成了一种 “间接继承” 的关系。 也就意味着最终的派生类在构造函数初始化中,要在初始化表中调用虚基类的构造函数进行初始化。...这样,就保证了虚基类不会被二次初始化。...private privat privat private private 另外,虚继承概念:【Example】C++ 虚基类与虚继承 (菱形继承问题) 虚继承时,子类的内存结构当中不包含父类。...【Example】C++ 接口(抽象类)概念讲解及例子演示 【Example】C++ 虚基类与虚继承 (菱形继承问题) 【Example】C++ Template (模板)概念讲解及编译避坑 【Example
菱形继承 菱形继承的概念 两个派生类继承同一个基类,又有某个类同时继承着这两个派生类 菱形继承典型案例 这种继承带来的问题主要有两方面: 羊和驼都继承了动物的类成员,当羊驼想要使用时,会产生二义性...m_Age,通过限定作用域的方式无法彻底解决这个问题,这个时候就要使用虚继承 虚继承与虚基类 具体实现为在羊类和驼类的继承前加上virtual关键词,Animal类称为虚基类 代码如下: #include... using namespace std; class Animal //虚基类 { public: int m_Age; }; class Sheep :virtual public...可以看出羊类和驼类中的数据只是一个虚基类指针,并未继承具体的数据,这个虚基类指针指向各自的虚基类表,而虚基类表中存在一个偏移量,通过这个偏移量再加上首地址可以找到基类中的数据,所以实际上羊驼只继承了一份数据...(也就是基类中的那份)。
所以基类的成员函数反正都要被覆盖,从某种意义上来基类的成员函数可以用纯虚函数来代替。...在C++语言中,基类必须将它的两种成员函数区分开来: 一种是基类希望其派生类进行覆盖的函数 另一种是基类希望派生类直接继承而不要改变的函数。 对于前者,基类通常将其定义为虚函数(virual)。...任何构造函数之外的非静态函数都可以是虚函数。 关键字virtual只能出现在类内部的声明语句之前而不能用于类外部的函数定义。 如果基类把一个函数声明成虚函数,则该函数在派生类中也是虚函数。...成员函数如果没被声明为虚函数,则其解析过程发生在编译时而非运行时。就会按照实际情况调用。 派生类可以继承定义在基类中的成员,但是派生类的成员函数不一定有权访问从基类继承而来的成员。...如果派生类没有覆盖其基类中的某个虚函数,则该虚函数的行为类似于其他的普通成员,派生类会直接继承其在基类中的版本,派生类可以在它覆盖的函数前使用virtual关键字,但不是非得这么做(可有可无)。
C.137: Use virtual bases to avoid overly general base classes C.137: 使用虚基类避免过于一般的基类 Reason(原因)...避免将所有的共享数据放进一个终极基类中。...如果很多派生类之间分享特别有用的共通的"实现细节",从中分离出共通功能就是有意义的。
二、虚继承与虚基类 当派生类从多个基类派生,而这些基类又从同一个基类派生,则在访问此共同基类中的成员时,将产生二义性,可以采用虚基类来解决。...从输出可以总结出: 1、虚基类的成员是由最远派生类的构造函数通过调用虚基类的构造函数进行初始化的。...2、在整个继承结构中,直接或间接继承虚基类的所有派生类,都必须在构造函数的成员初始化表中给出对虚基类的构造函数的调用。如果未列出,则表示调用该虚基类的默认构造函数。...3、在建立对象时,只有最远派生类的构造函数调用虚基类的构造函数,该派生类的其他基类对虚基类构造函数的调用被忽略。...参考: C++ primer 第四版 Effective C++ 3rd C++编程规范
这个要提一下,如果记不住就记住:如果不做虚析构函数,会有内存泄漏 解释 定义一个基类的指针p,在delete p时,如果基类的析构函数是虚函数,这时只会看p所赋值的对象,如果p赋值的对象是派生类的对象,...就会调用派生类的析构函数;如果p赋值的对象是基类的对象,就会调用基类的析构函数,这样就不会造成内存泄露。...如果基类的析构函数不是虚函数,在delete p时,调用析构函数时,只会看指针的数据类型,而不会去看赋值的对象,这样就会造成内存泄露。 多少学点设计模式就清楚了。...Inherit :public Base{ //此处省去,一切从简 }; //重点看调用 int main() { Base *p = new Inherit; //这种方式的调用,这时候有没有虚析构就不一样了
[TOC]Golang面向对象编程之继承&虚基类【组合&接口】201808相关说明Golang里面没有像C++一样有继承相关的概念,但是我们却可以实现继承相关的用法,这就要用到struct、interface...encrypt)) }实现继承直接上代码如下,很简单,主要就是一个struct里面包含一个匿名的struct,也就是通过匿名组合来实现package mainimport ( "fmt")// 【基类...,那么通过子类设置和获取得到的变量都是基类的 msgId int}func (group *GroupMsgModel) GetId() int { return group.msgId}/...fmt.Println("group.msgType =", group.msgType, "\tgroup.MsgModel.msgType =", group.MsgModel.msgType)}实现虚基类的用法...Golang可以interface + struct来实现虚基类的用法,必须要实现interface中定义的方法。1,定义一个interface接口MsgModel,包含了一些方法。
1.GNU C++平台 GNU C++平台获取C++成员虚函数地址可使用如下方法[1]^{[1]}[1]: class Base { int i; public: virtual void...2.Visual C++平台 可以采用内联汇编的方式获取[2]^{[2]}[2],代码如下: #define ShowFuncAddress(function) _asm{\ mov eax,...3.通过访问虚函数表获取虚函数地址 下面的代码可以在GCC和Visual C++共同编译运行。.../********************** @className:类名称 @pObj:类对象地址 @index:虚函数表项(从0开始) **********************/ void showVtableContent...Base's vtable[1]:0x00C811DB ---- 参考文献 [1]print address of virtual member function [2]C++动态联编实现原理分析
1.GCC平台 GCC平台获取C++成员虚函数地址可使用如下方法[1]^{[1]}: class Base{ int i; public: virtual void f1(){...2.Visual C++平台 可以采用内联汇编的方式获取[2]^{[2]},代码如下: #define ShowFuncAddress(function) _asm{\ mov eax, function...3.通过访问虚函数表获取虚函数地址 下面的代码可以在GCC和Visual C++共同编译运行。.../********************** @className:类名称 @pObj:类对象地址 @index:虚函数表项(从0开始) **********************/ void showVtableContent...f2(){ cout<<"Base's f2()"<<endl; } }; Base b; showVtableContent("Base",&b,0); //输出第一个虚函数
“强制实现”,也就是只要是父类指针调用普通函数,那就是父类的普通函数 而虚函数的作用,主要是为了让父类指针可以调用子类的函数,这种是在运行时才决定调用哪个函数 1、虚函数: C++的虚函数主要作用是...“运行时多态”,父类中提供虚函数的实现,为子类提供默认的函数实现。...子类可以重写父类的虚函数实现子类的特殊化。 2、纯虚函数: C++中包含纯虚函数的类,被称为是“抽象类”。抽象类不能使用new出对象,只有实现了这个纯虚函数的子类才能new出对象。 ...C++中的纯虚函数更像是“只提供申明,没有实现”,是对子类的约束,是“接口继承”。 C++中的纯虚函数也是一种“运行时多态”。...因此,在继承关系中,子类不应该重写父类的普通函数,因为函数的调用至于类对象的字面值有关。 参考链接
绿色碧玺 C.121: If a base class is used as an interface, make it a pure abstract class C.121:如果基类被用来定义接口...,保证它是一个纯虚类 Reason(原因) A class is more stable (less brittle) if it does not contain data....不包含数据的类会更稳定(更少脆弱性)。接口通常应该由公开的纯虚函数和默认/空的纯虚析构函数组成。...派生类通过它的Goof接口被销毁,(但是由于Goof的析构函数不是虚函数,导致Derived的析构函数不会被调用,译者注)因此它的string成员会发生泄露。...对于所有同时包含数据成员和并非继承自基类的可覆盖(非最终)虚函数的类发出警告。
在基类的定义中,定义虚函数的一般形式为: virtual 函数返回值类型 虚函数名(形参表) { 函数体 } 为什么说虚函数是C++最重要的特性之一呢,因为虚函数承载着C++中动态联编的作用,也即多态...1.2虚析构函数 在类的继承中,基类的析构函数一般都是虚函数。当基类中有虚函数的时候,析构函数也要定义为虚析构函数。...而且有时在编写基类的时候,发生了如下情况: (1)功能不应由基类去完成; (2)还没想好应该如何写基类的这个函数; (3)有的时候基类本身不应被实例化。 这时就可以用到纯虚函数了。...下面看一下纯虚函数与抽象类的实例。本实验在GNU C++环境下进行。...实现了基类Base类的虚函数和纯虚函数,同时注意到,企图去实例化抽象类,编译器会报错。
(覆盖):派生类中有一个跟基类完全相同的虚函数(即派生类虚函数与基类虚函数的返回值类型、函数名字、参数列表完全相同),称子类的虚函数重写了基类的虚函数 class Person { public: virtual...: 2.3.1.2 协变 (基类与派生类虚函数返回值类型不同) 派生类重写基类虚函数时,与基类虚函数返回值类型不同。...即基类虚函数返回基类对象的指针或者引用,派生类虚函数返回派生类对象的指针或者引用时,称为协变。...nullptr 总结一下派生类的虚表生成:a.先将基类中的虚表内容拷贝一份到派生类虚表中 b.如果派生类重写了基类中某个虚函数,用派生类自己的虚函数覆盖虚表中基类的虚函数 c.派生类自己新增加的虚函数按其在派生类中的声明次序增加到派生类虚表的最后...所以菱形继承、菱形虚拟继承我们的虚表我们就不看了,一般我们也不需要研究清楚,因为实际中很少用 C++ 虚函数表解析 | 酷 壳 - CoolShell C++ 对象的内存布局 | 酷 壳 - CoolShell
1.2 多态的构成条件 必须通过基类的指针或者引用调用虚函数 被调用的虚函数必须构成派生类对基类的重写(覆盖) 多态代码(完整版)示例: #include using namespace...void BuyTicket() { cout << "买票-半价" << endl; } }; 3.协变(基类与派生类虚函数返回值类型不同) 派生类重写基类虚函数时,与基类虚函数返回值类型不同...) 如果基类的析构函数为虚函数,此时派生类析构函数只要定义,无论是否加virtual关键字,都与基类的析构函数构成重写,虽然基类与派生类析构函数名字不同。...基类和派生类不会共用一张虚函数表,而是会依据基类自己构造来避免调用不明的问题 b. 同一个自定义类型的对象将会共用一张虚函数表 c....先将基类中的虚表内容拷贝一份到派生类虚表中 2. 然后如果派生类重写了基类中某个虚函数,用派生类自己的虚函数覆盖虚表中基类的虚函数 3.
在多态中,通常父类中虚函数的实现是毫无意义的,主要都是调用子类重写的内容。 因此可以将虚函数改为纯虚函数。...純虚函数语法:virtual 返回值类型 函数名(参数列表)=0; 当类中有了純虚函数,这个类也被称为抽象类。 抽象类特点: 无法实例化对象; 子类必须重写抽象类中的纯虚函数,否则也属于抽象类。
面试题:基类的析构函数为何要声明为虚函数? 在 C++ 中,一个类的析构函数用于释放它的实例占用的资源。如果没有正确地释放这些资源,就可能会导致内存泄漏和其他严重的问题。...基类的析构函数到底是否需要声明为虚函数取决于你是否会使用继承体系。 当使用继承时,如果一个基类指针指向了一个派生类对象,当对指针进行 delete 操作时,应该同时调用基类和派生类的析构函数。...如果基类的析构函数不是虚拟函数,则运行时不能确认要调用哪个析构函数。如果它是虚拟的,则始终将调用最实际对象的析构函数。...在 main() 函数中,通过基类指针创建一个 Dog 类型的对象,然后删除指针。...总结 在使用继承时,应该将基类的析构函数声明为虚函数,这样可以确保在运行时删除派生类对象时同时调用基类和派生类的析构函数。否则运行时不能确认要调用哪个析构函数,并且可能导致内存泄漏和其他问题。
领取专属 10元无门槛券
手把手带您无忧上云