首页
学习
活动
专区
工具
TVP
发布
精选内容/技术社群/优惠产品,尽在小程序
立即前往

基类派生类多态虚函数?

所以基类的成员函数反正都要被覆盖,从某种意义上来基类的成员函数可以用纯虚函数来代替。...任何构造函数之外的非静态函数都可以是虚函数。 关键字virtual只能出现在类内部的声明语句之前而不能用于类外部的函数定义。 如果基类把一个函数声明成虚函数,则该函数在派生类中也是虚函数。...成员函数如果没被声明为虚函数,则其解析过程发生在编译时而非运行时。就会按照实际情况调用。 派生类可以继承定义在基类中的成员,但是派生类的成员函数不一定有权访问从基类继承而来的成员。...此外,我们能将公有派生类型的对象绑定到基类的引用或指针上。 大多数类都只继承自一个类,这种形式的继承被称作“单继承”。 派生类中的虚函数派生类经常(但不总是)覆盖它继承的虚函数。...如果派生类没有覆盖其基类中的某个虚函数,则该虚函数的行为类似于其他的普通成员,派生类会直接继承其在基类中的版本,派生类可以在它覆盖的函数前使用virtual关键字,但不是非得这么做(可有可无)。

19520

C++中虚基类

---- 摘自《C++程序设计》 如果一个派生类有多个直接基类,而这些直接基类又有一个共同的基类,则在最终的派生类中会保留该间接共同基类数据成员的多份同名成员。...C++提供虚基类(virtual base class)的方法,使得在继承间接共同基类时只保留一份成员。 ---- 下面举例说明: 在如下的图中: ?...Person类是Student和Teacher的基类,而Graduate类又继承自Student和Teacher类。...但是显示,这我不是我们希望的,同样的副本我们只需要一份。所以C++中提出了虚基类的实现方式。...声明虚基类的一般形式是: class 派生类名:virtual 继承方式 基类名称 下面是上面实例的代码: 类声明person.h: #pragma once #include

64810
  • 您找到你想要的搜索结果了吗?
    是的
    没有找到

    派生类对基类中虚函数和非虚函数的继承效果

    而虚函数的作用,主要是为了让父类指针可以调用子类的函数,这种是在运行时才决定调用哪个函数 1、虚函数:   C++的虚函数主要作用是“运行时多态”,父类中提供虚函数的实现,为子类提供默认的函数实现。...子类可以重写父类的虚函数实现子类的特殊化。 2、纯虚函数:   C++中包含纯虚函数的类,被称为是“抽象类”。抽象类不能使用new出对象,只有实现了这个纯虚函数的子类才能new出对象。   ...C++中的纯虚函数更像是“只提供申明,没有实现”,是对子类的约束,是“接口继承”。   C++中的纯虚函数也是一种“运行时多态”。...3、普通函数:   普通函数是静态编译的,没有运行时多态,只会根据指针或引用的“字面值”类对象,调用自己的普通函数。   普通函数是父类为子类提供的“强制实现”。   ...因此,在继承关系中,子类不应该重写父类的普通函数,因为函数的调用至于类对象的字面值有关。 参考链接

    9210

    c++继承 基类 派生类 虚函数

    参考链接: C++继承 继承    类和类的关系有组合、继承和代理。继承的本质就是代码复用。子类继承父类中的一些东西,父类也称为基类,子类也称为派生类。派生类继承了基类除构造函数以外的所有成员。 ...1.调用基类的构造函数 2.调用派生类的构造函数 派生类的析构可想而知: 1.调用派生类的析构函数 2.调用基类的析构函数  虚函数  如下程序:  class Base { public:     Base...基类中含有虚函数,那么基类布局中存在一个虚函数指针,指向虚函数表;且其派生类中与其同名同参的函数不需要加virtual也是虚函数。...vfptr指针指向的vftable(虚函数表)中,&Base_meta中存放了RTTI信息(运行时类型信息),也就是class Base,0表示偏移,&Base::Show表示虚函数的入口地址。...首先通过指针所指向的对象找到vfptr,再找到vftable,获取到Show函数的入口地址,此时 &Derive::Show中存放的是派生类的虚函数入口地址,因此调用的是派生类中的Show()函数。

    1.1K20

    C++虚基类

    如果一个派生类有多个直接基类,而这些直接基类又有一个共同的基类,则在最终的派生类中会保留该间接共同基类数据成员的多份同名成员。C++提供虚基类的方法,使得在继承间接共同基类时只保留一份成员。...现在,将类A声明为虚基类,方法如下: class A//声明基类A {…}; class B :virtual public A//声明类B是类A的公用派生类,A是B的虚基类 {…}; class...C :virtual public A//声明类C是类A的公用派生类,A是C的虚基类 {…}; 注意: 虚基类并不是在声明基类时声明的,而是在声明派生类时,指定继承方式时声明的。...因为一个基类可以在生成一个派生类时作为虚基类,而在生成另一个派生类时不作为虚基类。...可以看到A的构造函数被调用了两遍!下面我们把A改成虚基类再看看结果。

    1.1K20

    面试题:基类的析构函数为何要声明为虚函数?

    面试题:基类的析构函数为何要声明为虚函数? 在 C++ 中,一个类的析构函数用于释放它的实例占用的资源。如果没有正确地释放这些资源,就可能会导致内存泄漏和其他严重的问题。...基类的析构函数到底是否需要声明为虚函数取决于你是否会使用继承体系。 当使用继承时,如果一个基类指针指向了一个派生类对象,当对指针进行 delete 操作时,应该同时调用基类和派生类的析构函数。...如果基类的析构函数不是虚拟函数,则运行时不能确认要调用哪个析构函数。如果它是虚拟的,则始终将调用最实际对象的析构函数。...在 main() 函数中,通过基类指针创建一个 Dog 类型的对象,然后删除指针。...总结 在使用继承时,应该将基类的析构函数声明为虚函数,这样可以确保在运行时删除派生类对象时同时调用基类和派生类的析构函数。否则运行时不能确认要调用哪个析构函数,并且可能导致内存泄漏和其他问题。

    4700

    C++编程经验(2):为虚基类做虚析构函数的必要性

    这个要提一下,如果记不住就记住:如果不做虚析构函数,会有内存泄漏 解释 定义一个基类的指针p,在delete p时,如果基类的析构函数是虚函数,这时只会看p所赋值的对象,如果p赋值的对象是派生类的对象,...就会调用派生类的析构函数;如果p赋值的对象是基类的对象,就会调用基类的析构函数,这样就不会造成内存泄露。...如果基类的析构函数不是虚函数,在delete p时,调用析构函数时,只会看指针的数据类型,而不会去看赋值的对象,这样就会造成内存泄露。 多少学点设计模式就清楚了。...接下来是一个子类 class Inherit :public Base{ //此处省去,一切从简 }; //重点看调用 int main() { Base *p = new Inherit; //这种方式的调用...,这时候有没有虚析构就不一样了 delete p; Base *q = new Base; delete q; return 0; }

    59710

    【c++】多态&&虚函数&&抽象类&&继承中的虚函数表详解

    那么在继承中要构成多态还有两个条件: 必须通过基类的指针或者引用调用虚函数 被调用的函数必须是虚函数,且派生类必须对基类的虚函数进行重写 2.2 虚函数 虚函数:即被virtual修饰的类成员函数称为虚函数...(覆盖):派生类中有一个跟基类完全相同的虚函数(即派生类虚函数与基类虚函数的返回值类型、函数名字、参数列表完全相同),称子类的虚函数重写了基类的虚函数 class Person { public: virtual...即基类虚函数返回基类对象的指针或者引用,派生类虚函数返回派生类对象的指针或者引用时,称为协变。...(基类与派生类析构函数的名字不同) 如果基类的析构函数为虚函数,此时派生类析构函数只要定义,无论是否加virtual关键字,都与基类的析构函数构成重写,虽然基类与派生类析构函数名字不同。...nullptr 总结一下派生类的虚表生成:a.先将基类中的虚表内容拷贝一份到派生类虚表中 b.如果派生类重写了基类中某个虚函数,用派生类自己的虚函数覆盖虚表中基类的虚函数 c.派生类自己新增加的虚函数按其在派生类中的声明次序增加到派生类虚表的最后

    38510

    C++虚拟继承与虚基类

    (1)当在多条继承路径上有一个公共的基类,在这些路径中的某几条汇合处,这个公共的基类就会产生多个实例(或多个副本),若只想保存这个基类的一个实例,可以将这个公共基类说明为虚基类,就像Student和Employee...(2)被虚拟继承的基类,叫做虚基类。虚基类实际指的是继承的方式,而非一个基类,是动词,而非名词。 (3)为了实现虚拟继承,派生类对象的大小会增加4。...这个增加的4个字节,是因为当虚拟继承时,无论是单虚继承还是多虚继承,派生类需要有一个虚基类表来记录虚继承关系,所以此时子类需要多一个虚基类表指针,而且只需要一个即可。...(4)虚拟继承中,虚基类对象是由最远派生类的构造函数通过调用虚基类的构造函数进行初始化的,派生类的构造函数的成员初始化列表中必须列出对虚基类构造函数的调用;如果未列出,则表示使用该虚基类的缺省构造函数。...因为虚拟继承只是表名某个基类的对象在派生类对象中只被构造一次,而在本例中类Student和Employee对象在EGStudent对象中本来就不会被构造多次,所以不将它们申明虚基类也是完全可以的。

    91120

    【Example】C++ 虚基类与虚继承 (菱形继承问题)

    这时,Base 便成了 Byte 和 Expert 的虚基类,达成了虚继承的方式,Base 类在最终的 Blu 类中只存在一个,所以不存在命名空间冲突及资源浪费。...虚基类并不是“绝对的”,而是“相对的”:虚基类在它自身声明、定义的时候无需任何修饰,只是在子类继承时进行 virtual 修饰。...所以在 Blu 类中仍然存在菱形继承的问题,所有需要将所有继承同一基类的上级父类继承方式声明为 virtual。...同时,在虚继承机制当中,虚基类是由最终的派生类进行初始化的,本身达成了一种 “间接继承” 的关系。 也就意味着最终的派生类在构造函数初始化中,要在初始化表中调用虚基类的构造函数进行初始化。...: Expert() {}; ~Expert() {}; }; class Blu : public Byte, public Expert { public: // 调用虚基类构造函数

    1K30

    【C++】多态 ⑧ ( 验证指向 虚函数表 的 vptr 指针 | 对比定义了虚函数的类和没有定义虚函数类的大小 )

    1 个虚函数 ; 如果 没有虚函数 , 就不会生成虚函数表 ; 如果 类 中有 virtual 虚函数 , 则 该类的 每个对象 中 , 都有一个 指向 虚函数表的 vptr 指针 ; 虚函数表 存储...虚函数指针 : " 虚函数表 " 是 存储 " 类成员函数指针 " 的 数据结构 , 是一个 函数指针数组 , 数组中的元素都是函数指针 , 具体存储的都是 指向 类中的虚函数 的指针 ; 如果 子类...中 , 重写了 父类的 virtual 虚函数 , 那么 C++ 编译器会在 子类 虚函数表 中放入该 子类虚函数的 函数指针 ; 如果 C++ 类中存在 virtual 虚函数 , 在创建对象时 ,...; 2、虚函数类与普通函数类对比 - 多出了 vptr 指针的大小 下面的代码中 , 定义了 2 个类 , 区别是 一个定义了 virtual 虚函数 , 另外一个没有定义 虚函数 ; 在 Parent...中定义了 虚函数 virtual void fun(int a) ; 在 Parent2 中定义的是 普通函数 void fun(int a) ; 使用 sizeof 函数 , 获取这两个类的大小 ,

    22740

    菱形继承问题及解决方法—虚继承与虚基类(C++)

    菱形继承 菱形继承的概念 两个派生类继承同一个基类,又有某个类同时继承着这两个派生类 菱形继承典型案例 这种继承带来的问题主要有两方面: 羊和驼都继承了动物的类成员,当羊驼想要使用时,会产生二义性...很明显羊驼从羊和驼两个父类中各自继承了一份m_Age,通过限定作用域的方式无法彻底解决这个问题,这个时候就要使用虚继承 虚继承与虚基类 具体实现为在羊类和驼类的继承前加上virtual关键词,...Animal类称为虚基类 代码如下: #include using namespace std; class Animal //虚基类 { public: int m_Age...可以看出羊类和驼类中的数据只是一个虚基类指针,并未继承具体的数据,这个虚基类指针指向各自的虚基类表,而虚基类表中存在一个偏移量,通过这个偏移量再加上首地址可以找到基类中的数据,所以实际上羊驼只继承了一份数据...(也就是基类中的那份)。

    1.2K40

    虚析构函数? vptr? 指针偏移?多态数组? delete 基类指针 内存泄漏?崩溃?

    2、在遇到通过基类指针或引用调用虚函数的语句时,首先根据指针或引用的静态类型来判断所调函数是否属于该class或者它的某个public 基类,如果 属于再进行调用语句的改写: (*(p->_vptr[slotNum...5、 在类继承机制中,构造函数和析构函数具有一种特别机制叫 “层链式调用通知” 《 C++编程思想 》 C++标准规定:基类的析构函数必须声明为virtual, 如果你不声明,那么"层链式调用通知"这样的机制是没法构建起来...,这是因为我们将基类的析构函数声明为虚函数的原因,在pI 指向派生类首地址的前提下,如果~IRectangle()  是虚函数,那么会找到实际的函数~Rectangle() 执行,而~Rectangle...将基类析构函数改成虚函数,fun() 最好也改成虚函数,只要有一个虚函数,基类大小就为一个vptr ,此时基类和派生类大小都是4个字节,p也指向派生类的首地址,问题解决,参考规则3。...也是论坛上经常讨论的,也就是说delete 基类指针(在指针没有偏离的情况下) 会不会造成内存泄漏的问题,上面说到如果此时基类析构函数为虚函数,那么是不会内存泄漏的,如果不是则行为未定义。

    98700

    虚析构函数? vptr? 指针偏移?多态数组? delete 基类指针 内存泄漏?崩溃?

    2、在遇到通过基类指针或引用调用虚函数的语句时,首先根据指针或引用的静态类型来判断所调函数是否属于该class或者它的某个public 基类,如果 属于再进行调用语句的改写: (*(p->_vptr[slotNum...5、 在类继承机制中,构造函数和析构函数具有一种特别机制叫 “层链式调用通知” 《 C++编程思想 》 C++标准规定:基类的析构函数必须声明为virtual, 如果你不声明,那么"层链式调用通知"这样的机制是没法构建起来...,这是因为我们将基类的析构函数声明为虚函数的原因,在pI 指向派生类首地址的前提下,如果~IRectangle() 是虚函数,那么会找到实际的函数~Rectangle() 执行,而~Rectangle...将基类析构函数改成虚函数,fun() 最好也改成虚函数,只要有一个虚函数,基类大小就为一个vptr ,此时基类和派生类大小都是4个字节,p也指向派生类的首地址,问题解决,参考规则3。...也是论坛上经常讨论的,也就是说delete 基类指针(在指针没有偏离的情况下) 会不会造成内存泄漏的问题,上面说到如果此时基类析构函数为虚函数,那么是不会内存泄漏的,如果不是则行为未定义。

    1K20

    抽象类纯虚函数与虚析构

    纯虚函数,一般是在设计一个基类时使用的,它将接口函数设置为纯虚函数后,只提供子类去继承并实现,以形成多态,除此以外不提供任何其他功能,我们称这种类为抽象类(abstract)。...【纯虚函数的声明语法】 virtual 函数声明 = 0; 【纯虚函数的特征】 1、只有声明,没有实现代码。 2、含有纯虚函数的类称为抽象类(abstract)。不能被实例化,只能被继承。...3、继承抽象类的意义就是提供族类的公共接口。 4、子类继承的纯虚函数,如果未实现,子类仍然为抽象类,仍然不能被实例化。 【虚函数的若干限制】 1、只有类的成员函数才能声明为虚函数。...虚函数仅适用于有继承关系的类对象,所以普通函数不能声明为虚函数。 2、静态成员函数不能是虚函数静态成员函数不受对象的捆绑,只有类的信息。 3、内联函数不能是虚函数。...4、构造函数不能是虚函数,构造时,对象的创建尚未完成。构造完成后,能算一个名符其实的对象。 5、析构函数可以是虚函数且通常声明为虚函数。

    20730

    Golang面向对象编程之继承&虚基类【组合&接口】

    [TOC]Golang面向对象编程之继承&虚基类【组合&接口】201808相关说明Golang里面没有像C++一样有继承相关的概念,但是我们却可以实现继承相关的用法,这就要用到struct、interface...var RsaSecuritySrv rsaSecurity类似于构造函数的定义,也可以通过new一个对象来使用,二选一。...,很简单,主要就是一个struct里面包含一个匿名的struct,也就是通过匿名组合来实现package mainimport ( "fmt")// 【基类】//定义一个最基础的struct类MsgModel...fmt.Println("group.msgType =", group.msgType, "\tgroup.MsgModel.msgType =", group.MsgModel.msgType)}实现虚基类的用法...Golang可以interface + struct来实现虚基类的用法,必须要实现interface中定义的方法。1,定义一个interface接口MsgModel,包含了一些方法。

    1.9K31

    虚函数与纯虚函数的区别

    如下就是一个父类中的虚函数: class A { public: virtual void ss() { cout基类的虚函数"<<endl; } }...; 纯虚函数(pure virtual)   C++中包含纯虚函数的类,被称为是“抽象类”。...抽象类不能使用new出对象,只有实现了这个纯虚函数的子类才能new出对象。   C++中的纯虚函数更像是“只提供申明,没有实现”,是对子类的约束,是“接口继承”。   ...} //虚函数 virtual void xhs(){ //这个虚函数必须得在基类中实现 cout基类的虚函数"的虚函数也要在基类中实现...} //派生类中可以不写这个函数,但是派生类对象调用时会调用积累的虚函数 //纯虚函数 virtual void cxhs() =0; //这个纯虚函数不在基类中实现,必须在子类中实现

    1.3K10

    C++核心准则C.35:基类的析构函数要么是公开的虚函数,要么是保护的非虚函数

    为了避免无定义的行为。如果析构函数是公有的,那么调用侧的代码就会尝试使用基类指针销毁派生类的对象,在基类的析构函数为非虚函数时其结果时没有定义的。...如果析构函数时保护的,那么调用侧代码就无法通过基类类型指针销毁派生类对象,这是析构函数就没有必要一定是虚函数。析构函数是保护而不是私有的,这样派生类的析构函数才能调用它。...通常,基类的设计者不会知道在析构函数中应该执行什么样的动作。...我们可以想象一种需要保护的虚函数析构函数的情况:当希望允许派生类的对象(只有这个类型)通过基类指针销毁另外一个对象(不是它自己)时。但是我们还没有在实际的开发中遇到这种情况。...拥有虚函数的类的虚函数要么是公开的虚函数,要么是保护的非虚函数。 译者注:拥有虚函数一般就意味着它有派生类。

    1.1K20
    领券