函数模板的使用 动态多态 在基类的函数前加上virtual关键字,在派生类中重写该函数,运行时将会根据所指对象的实际类型来调用相应的函数,如果对象类型是派生类,就调用派生类的函数,如果对象类型是基类,...2、纯虚函数是在基类中声明的虚函数,它在基类中没有定义,但要求任何派生类都要定义自己的实现方法。在基类中实现纯虚函数的方法是在函数原型后加"=0" 3、声明了纯虚函数的类是一个抽象类。...只有在基类析构函数定义为虚函数时,调用操作符delete销毁指向对象的基类指针时,才能准确调用派生类的析构函数(从该级向上按序调用虚函数),才能准确销毁数据。...2、虚函数表类似一个数组,类对象中存储vptr指针,指向虚函数表。...3、虚函数表存储虚函数的地址,即虚函数表的元素是指向类成员函数的指针,而类中虚函数的个数在编译时期可以确定,即虚函数表的大小可以确定,即大小是在编译时期确定的,不必动态分配内存空间存储虚函数表。
重载 overload,指对一个同名方法进行了几种不同参数的实现 可以将一个派生类对象转换为基类对象,此时派生类独有的部分将被截断,其基类部分被处理而派生类部分被忽略 有时我们不希望派生类独有的部分被截断则需要使用类指针来调用重载的函数或使用指针所指的成员...我们很多时候希望的是我们通过将基类指针指向派生类,然后可以动态调用派生类的函数,这时我们可以将基类的对应函数写为虚(virtual)函数来实现,此时发生的称为动态绑定 派生类可以继承多个基类,称为多继承...因此除了重载虚函数外最好不要让名称同名 派生类可以覆盖基类重载的函数,但是如果派生类希望基类重载的几个函数都在派生类中可见的话:一种方法是不覆盖任何一个重载函数或将所有重载函数都进行一次覆盖;另一种方法是为需要重载的函数名使用...using语句并不要指定形参列表,可以将所有重载函数都加入派生类的作用域中 继承体系中,最关键的是基类通常需要定义一个虚析构函数,这样我们才能动态分配体系中的对象,确保delete时能够执行正确的析构函数版本...,这些实参不会被继承,而是派生类会得到多个继承的构造函数,每个构造函数省略一个有默认实参的形参 当我们想要把继承体系的对象存放到容器中时,最好使用间接存储也就是存放基类指针(智能指针就更好了) 16
=demo2; //派生类对象可以初始化基类引用; Base *Bpo=&demo2;//派生类对象可以赋给指向基类对象的指针;//多态实现的方法 主要是派生类中一定包含基类中所有成员,在使用中,一定可以找到对应成员...赋值兼容应注意的问题: 指向基类的指针可以指向公有派生类的对象,但不允许指向它的私有派生类的对象。...允许将一个声明为指向基类的指针指向其公有派生类对象,但是不能将一个声明为指向派生类对象的指针指向基类对象。...声明为指向基类对象的指针,当其指向公有派生类对象时,只能用它来直接访问派生类中从基类继承来的成员,而不能直接访问公有派生类的定义的成员。...在编译的时候能够确定对象所调用的成员函数的地址则为静态联编,一般的调用方式; 动态联编:指的是在程序运行的时候动态地进行,根据当时的情况来确定调用哪个同名函数,父类指针指向哪个子类,就调用哪个子类的同名函数
继承与派生 声明派生类的一般形式为: class 派生类名 [继承方式] 基类名{ 派生类新增加的成员 }; 其中的继承方式包括 public (公用的)、 private (私有的)和 protected...同时,无论哪一种继承方式,在派生类中是不能访问基类的私有 成员的,私有成员只能被本类的成员函数所访问,毕竟派生类与基类不是同一个类 构造派生类的对象时,必须对基类数据成员、新增数据成员和成员对象的数据成员进行初始化...而 虚函数 的作用是允许在派生类中重新定义与基类同名的函数,并且可以通过基类指针或引用来访问基类和派生类中的同名函数。...虚函数的声明方式: virtual 返回类型 函数名(); 当把基类某个成员函数声明为虚函数后,就允许在其派生类中对该函数重新定义,赋予它新的功能,且可以通过指向基类的指针指向同一类族中不同类的对象,从而调用其中的同名函数...纯虚函数是在基类中声明的虚函数,它在基类中没有定义,但要求任何派生类都要定义自己的实现方法。
动态内存直接管理方式:new/delete,new在自由存储空间上分配对象内存,对象使用完毕需使用delete释放new分配的对象指针指定的内存空间;delete p释放对象,delete []释放p指向的数组...private继承方式基类中的所有 public 成员在派生类中均为 private 属性;基类中的所有 protected 成员在派生类中均为 private 属性;基类中的所有 private 成员在派生类中不能使用...,则pb指针对象使用的仍是基类指针对象,故调用的是基类的func函数。...每个虚继承的子类都有一个虚基类指针(占用一个指针的存储空间,4字节)和虚基类表(不占用类对象的存储空间)当派生类重新定义虚函数时,则将派生类的虚函数的地址添加到虚函数表中。...当一个基类指针指向一个派生类对象时,虚函数表指针指向派生类对象的虚函数表。当调用虚函数时,由于派生类对象重写了派生类对应的虚函数表项,基类在调用时会调用派生类的虚函数,从而产生多态。
虚函数 若派生类中定义了于基类相同的函数,使用基类指针指向派生类对象时,通过指针调用的是基类函数而不是派生类中的函数。...编译器的做法不是我们期望的,根据实际的对象类型来判断重写函数的调用,如果父类指针指向的是父类对象则调用父类中定义的函数,如果父类指针指向的是子类对象则调用子类中定义的重写函数。 ...纯虚函数是在基类中声明的虚函数,它在基类中没有定义,但要求任何派生类都要定义自己的实现方法。...同时抽象类只能作为基类来使用,其纯虚函数的实现由派生类给出。如果派生类中没有重新定义纯虚函数,而只是继承基类的纯虚函数,则这个派生类仍然还是一个抽象类。...如果派生类中给出了基类纯虚函数的实现,则该派生类就不再是抽象类了,它是一个可以建立对象的具体的类。
最常见的使用方法就是声明基类的指针,利用该指针指向随意一个子类对象,调用对应的虚函数,能够依据指向的子类的不同而实现不同的方法。...假设没有使用虚函数的话,即没有利用C++多态性,则利用基类指针调用对应的函数的时候,将总被限制在基类函数本身,而无法调用到子类中被重写过的函数。...p->foo()和p->fuu()则是基类指针指向子类对象,正式体现多态的使用方法,p->foo()因为指针是个基类指针,指向是一个固定偏移量的函数,因此此时指向的就仅仅能是基类的foo()函数的代码了...并非非常理解这样的使用方法,从原理上来解释,因为B是子类指针,尽管被赋予了基类对象地址,可是ptr->foo()在调用的时候,因为地址偏移量固定,偏移量是子类对象的偏移量,于是即使在指向了一个基类对象的情况下...C++纯虚函数 一、定义 纯虚函数是在基类中声明的虚函数,它在基类中未定义,但要求不论什么派生类都要定义自己的实现方法。
用继承的对类创建的对象可以调用父类的成员函数,但要注意,派生类继承了父类的成员函数和成员变量,不是拷贝了一份到派生类里,而是给了派生类去父类调用父类函数的权利。...基类private成员在派生类中无论以什么方式继承都是不可见的。这里的不可见是指基类的私有成员还是被继承到了派生类对象中,但是语法上限制派生类对象不管在类里面还是类外面 都不能去访问它。 2....一般都是使用public/protected方式进行继承 二.基类和派生类对象赋值转换 派生类对象可以赋值给基类的对象 / 基类的指针 / 基类的引用。这里有个形象的说法叫切片或者切割。...寓意把派生类中父类那部分切来赋值过去。 基类对象不能赋值给派生类对象。 基类的指针或者引用可以通过强制类型转换赋值给派生类的指针或者引用。但是必须是基类 的指针是指向派生类对象时才是安全的。...(在子类成员函数中,可以使用 基类::基类成员 显示访问) 3. 需要注意的是如果是成员函数的隐藏,只需要函数名相同就构成隐藏。 4. 注意在实际中在继承体系里面最好不要定义同名的成员。
使用虚函数实现多态行为 可通过Fish指针或Fish引用访问Fish对象,这种指针或引用可指向Fish、Carp等对象。但你不需要知道也不关心它们指向的是哪种对象。...为什么需要虚构函数 上面的代码如果加入析构函数释放内存,对于使用new在自由储存区中实例化的派生类对象,如果将其赋值给基类指针,并通过该指针调用delete,将不会调用派生类的析构函数,这可能导致资源未释放...new在自由存储区中实例化的,还是以局部变量的方式在栈中实例化的,构造函数和析构函数的调用顺序都相同。...抽象基类提供了一种非常好的机制,让您能够声明所有派生类都必须实现的函数。抽象基类常被简称为ABC。ABC有助于约束程序的设计。...注意:C++关键字virtual的含义随上下文而异(我想这样做的目的很可能是为了省事),对其含义总结如下: 在函数声明中,virtual意味着当基类指针指向派生对象时,通过它可调用派生类的相应函数。
假设该指针是一个指向派生类对象的基类类型指针,则那个对象将被切割,仅仅抛出基类部分。 谨记:抛出指向局部对象的指针总是错误的,因此,在抛出指针的时候,必须确定进入处理代码时指针所指向的对象存在。...3、异常说明符与继承 像形參声明一样,基类的异常说明符能够用于捕获派生类型的异常对象,并且,异常说明符的静态类型决定catch子句能够运行的动作。...假设被抛出的异常对象是派生类类型的,但由接受基类类型的catch处理,那么,catch不能使用派生类特有的不论什么成员。...假设catch对象是基类类型对象而异常对象是派生类型的,就将异常对象切割为它的基类子对象。 对象(相对于引用)不是多态的。对象的静态类型和动态类型相同,函数是虚函数也一样。...由于catch子句按出现次序匹配,所以使用来自继承层次的异常的程序将它们的catch子句排序,以便派生类型的处理代码出如今其基类类型的catch之前。
定义 纯虚函数就是在基类中声明的虚函数,它在基类中没有定义,但要求任何派生类都要定义自己的实现方法。...在基类中实现纯虚函数的方法是在函数原型后面加“=0” 引入原因 方便的使用多态,因此常常在基类中定义虚函数 在很多情况下,基类本身生成对象是不合理的。...使用抽象类时注意: 抽象类只能作为基类来使用,其纯虚函数的实现由派生类给出。如果派生类中没有重新定义纯虚函数,而只是继承基类的纯虚函数,则这个派生类仍然是一个抽象类。...包含纯虚函数的类是抽象类,抽象类不能定义实例,但可以声明指向实现该抽象类的具体类的指针或引用。...但可以通过让友元函数调用虚拟成员函数来解决友元的虚拟问题。 析构函数应当是虚函数,将调用相应对象类型的析构函数。因此,如果指针指向的是子类对象,将调用子类的析构函数,然后自动调用基类的析构函数。
33、为什么析构函数一般写成虚函数 由于类的多态性,基类指针可以指向派生类的对象,如果删除该基类的指针,就会调用该指针指向的派生类析构函数,而派生类的析构函数又自动调用基类的析构函数,这样整个派生类的对象完全被释放...只有在基类析构函数定义为虚函数时,调用操作符delete销毁指向对象的基类指针时,才能准确调用派生类的析构函数(从该级向上按序调用虚函数),才能准确销毁数据。...在构造子类对象时,会先调用父类的构造函数,此时,编译器只“看到了”父类,并为父类对象初始化虚表指针,令它指向父类的虚表;当调用子类的构造函数时,为子类对象初始化虚表指针,令它指向子类的虚表 (4)当派生类对基类的虚函数没有重写时...,派生类的虚表指针指向的是基类的虚表;当派生类对基类的虚函数重写时,派生类的虚表指针指向的是自身的虚表;当派生类中有自己的虚函数时,在自己的虚表中将此虚函数地址添加在后面 这样指向派生类的基类指针在运行时...上述观点看似正确,其实不然,如果虚函数在编译器就能够决定将要调用哪个函数时,就能够内联,那么什么情况下编译器可以确定要调用哪个函数呢,答案是当用对象调用虚函数(此时不具有多态性)时,就内联展开 综上,当是指向派生类的指针
派生类与基类之间的关系: 派生类对象可以使用基类(公有的)方法。 基类指针可以在不进行显示类型转换的情况下指向派生类对象,但只能调用基类方法。...且该转换是可以传递的,例如基类A,其派生类AP,AP的派生类APP,则A指针或引用可以指向或引用AP类对象和APP类对象。 相反的,我们将基类指针或引用转换为派生类指针或引用称为向下强制转换。...但该种转换只能使用显示类型转换,防止无意间指向派生类独有的方法或成员造成的异常情况的发生。...虚函数表存储了为类对象进行声明的虚函数的地址。通常情况下,基类对象包含一个指向该类中虚函数表的指针。...因此,如果要重新定义继承的方法,则应确保与原来的原型完全相同,但是如果返回类型是基类引用或指针,则可以修改为指向派生类的引用或指针,即允许返回类型随类类型的变化而变化,这种特性被称为返回类型协变。
任何构造函数以外的非静态函数都可以是虚函数,如果基类声明了虚函数,那么派生类中的对应函数都隐式的是虚函数 通过抽象,我们使用动态绑定可以实现接口与实现的分离,基类用虚函数声明出接口,然后指针指向不同的派生类实现来动态调用...如果是指针或引用的转换,则只是一个指向的改变 派生类的构造函数需要负责所有成员的初始化,尽管派生类也可以初始化继承来的基类成员,但是这不符合通常的编码思路,派生类一般在构造函数开始的地方调用基类的构造函数...这里有一个特别的,即便处理的是基类指针,此指针指向某派生类,我们也不能隐式转换到这个派生类,如果基类中含有虚函数,我们可用用dynamic_cast强制转换 15.3 虚函数 通过对基类的指针或引用来调用虚函数时会出现动态绑定...,对于实现的内容我们一样可以使用=default简化 如果基类的析构函数不是虚函数,则delete一个指向派生类对象的基类指针将产生未定义的行为 如果定义了虚析构函数,则一样的合成移动操作将被阻止 派生类的析构函数和以往一样是空函数...,最好使用间接存储也就是存放基类指针(智能指针就更好了),这是因为之前说到的截断特性导致的 我们常常定义额外的容器类来保存类的指针们 有时我们需要写好几个虚函数来处理容器中的多态问题 当我们继承一个类时
在派生类对象中,包含着基类对象,而且基类对象的存储位置位于派生类对象新增的成员变量之前,相当于基类对象是头部。...对象 B 不但 A 对象能访问,其他需要用它的对象也可以使用。 指针对象可以使用多态的特性,基类的指针可以指向派生链的任意一个派生类。...第33-36行代码是正确派生类构造函数初始化基类构造函数的方式,通过调用基类构造函数来初始化基类,在执行一个派生类的构造函数 之前,总是先执行基类的构造函数。...所以派生方式是 private 或 protected,则是无法像 public 派生承方式一样把派生类对象赋值、引用、指针给基类对象。...---- || 03 基类与派生类的指针强制转换 public 派生方式的情况下,派生类对象的指针可以直接赋值给基类指针: Base *ptrBase = & objDerived; ptrBase 指向的是一个
虚函数的作用是允许在派生类中重新定义与基类同名的函数,并且可以通过基类指针或引用来访问基类和派生类中的同名函数。 请分析例12.2。这个例子开始时没有使用虚函数,然后再讨论使用虚函数的情况。...如果能够用同一种方式去调用同一类族中不同类的所有的同名函数,那就好了。 用虚函数就能顺利地解决这个问题。...说明:本来基类指针是用来指向基类对象的,如果用它指向派生类对象,则进行指针类型转换,将派生类对象的指针先转换为基类的指针,所以基类指针指向的是派生类对象中的基类部分。...如果在派生类中没有对基类的虚函数重新定义,则派生类简单地继承其直接基类的虚函数。 定义一个指向基类对象的指针变量,并使它指向同一类族中需要调用该函数的对象。...通过该指针变量调用此虚函数,此时调用的就是指针变量指向的对象的同名函数。 通过虚函数与指向基类对象的指针变量的配合使用,就能方便地调用同一类族中不同类的同名函数,只要先用基类指针指向即可。
多态的表现形式一 「派生类的指针」可以赋给「基类指针」; 通过基类指针调用基类和派生类中的同名「虚函数」时: 若该指针指向一个基类的对象,那么被调用是 基类的虚函数; 若该指针指向一个派生类的对象,那么被调用...是不是大家觉得 pBase 指针对象虽然指向的是派生类对象,但是派生类里没有 fun1 成员函数,则就调用基类的 fun1 成员函数,Base::fun1() 里又会调用基类的 fun2 成员函数,所以输出结果是...} } this 指针的作用就是指向成员函数所作用的对象, 所以非静态成员函数中可以直接使用 this 来代表指向该函数作用的对象的指针。...pBase 指针对象指向的是派生类对象,派生类里没有 fun1 成员函数,所以就会调用基类的 fun1 成员函数,在Base::fun1() 成员函数体里执行 this->fun2() 时,实际上指向的是派生类对象的...int a; }; 包含纯虚函数的类叫抽象类 抽象类只能作为基类来派生新类使用,不能创建抽象类的对象 抽象类的指针和引用可以指向由抽象类派生出来的类的对象 A a; // 错,A
对于C++程序员来说,精通指针的使用是非常重要的。 重写和重载的区别 重写指的是在派生类中重新定义基类的虚函数的行为。...当基类中的某个虚函数在派生类中被重新定义时,如果派生类对象调用该函数,则会覆盖掉基类中的实现,执行派生类中的实现代码。在进行对象的多态性转换时,重写非常重要。...C++多态是指在继承关系中,子类可以重写父类的虚函数,从而使得一个指向子类对象的指针能够调用子类的函数而不是父类的函数。其底层原理涉及到虚函数表、虚指针等概念。...基类的析构函数为何要声明为虚函数? C++基类的析构函数声明为虚函数是为了确保在通过基类的指针或引用删除派生类对象时,可以正确地释放派生类对象所占用的内存。...如果基类的析构函数不是虚函数,则在这种情况下只会调用基类的析构函数,而不会调用派生类的析构函数,从而可能导致内存泄漏和未定义行为。
派生类继承基类的访问控制标号【何种方式继承】无论是什么,不影响派生类使用基类成员,但影响使用派生类的用户访问基类成员。类使用接口继承还是实现继承对派生类用户具有重要含义。 友元关系不继承。...如果知道基类到派生类的转换【这种转换是基类地址赋给派生类指针】是安全的【就是说心里清楚基类指针指向的确实是派生类】,可以使用static_cast强制编译器进行转换。...派生类指针的静态类型和动态类型不一致时【基类指针指向派生类是时】,为保证删除指针调用合适的析构函数【多态】,基类析构必须为virtual。...构造函数是对象动态类型确定之前运行的,不需要定义为virtual。 引用、对象、指针的静态类型决定了能够完成的行为,动态类型有多的功能也无法使用。派生类应避免与基类成员名字冲突。...对象不支持动态绑定,指针和引用支持但使用起来麻烦,解决方法是定义包装类或句柄类【提供到其它类接口的类】。像使用指针一样使用句柄而不用管理它指向的对象。类似智能指针的方法建立指针句柄。
领取专属 10元无门槛券
手把手带您无忧上云