是多态性的一种应用,也称为向上转型。在面向对象的编程中,派生类是基类的扩展,通过将派生类的地址分配给基类指针,可以实现对派生类对象的访问和操作。
这种技术的优势在于可以通过基类指针来访问派生类的成员函数和成员变量,实现对派生类对象的多态操作。通过使用基类指针,可以方便地处理一组派生类对象,而无需关心具体的派生类类型。
应用场景:
推荐的腾讯云相关产品和产品介绍链接地址:
请注意,以上链接仅供参考,具体产品选择应根据实际需求进行评估和决策。
基类指针指向派生类,我们已经很熟了。假如我们想用派生类反过来指向基类,就需要有两个要求:1)马克-to-win:基类指针开始时指向派生类,2)我们还需要清清楚楚的转型一下。
.152: Never assign a pointer to an array of derived class objects to a pointer to its base C.152:永远不要将派生类数组的指针赋值给基类指针...作为赋值结果的基类指针的下标运算会引起无效的对象访问并可能发生内存破坏。...提示所有数组退化和基类类型向派生类类型转换的情况。...don't let the array name suffer a derived-to-base conversion before getting into the span 使用span传递数组而不是指针...,也不要再放入span之前让数组名经过一次派生类向基类类型的转换。
当创建一个类时,程序员不需要完全重新编写新的数据成员和成员函数,只需要设计一个新的类,继承了已有的类的成员即可。这个已有的类被称为的基类,这个新的类被称为派生类。...基类和派生类 一个类可以派生自多个类或接口,这意味着它可以从多个基类或接口继承数据和函数。...C# 中创建派生类的语法如下: class { ... } class : { ... } 假如一个基类是Shape,一个派生类是Rectangle...Console.WriteLine(rect.getArea()); Console.ReadLine(); } } } 结果: 30 基类的初始化...派生类继承了基类的成员变量和成员方法。
C++中派生类对基类成员的访问形式主要有以下两种: 1、内部访问:由派生类中新增成员对基类继承来的成员的访问。 2、对象访问:在派生类外部,通过派生类的对象对从基类继承来的成员的访问。...今天给大家介绍在3中继承方式下,派生类对基类成员的访问规则。...基类的private成员在私有派生类中是不可直接访问的,所以无论是派生类的成员还是通过派生类的对象,都无法直接访问从基类继承来的private成员,但是可以通过基类提供的public成员函数间接访问。...基类的private成员在私有派生类中是不可直接访问的,所以无论是派生类成员还是派生类的对象,都无法直接访问从基类继承来的private成员,但是可以通过基类提供的public成员函数直接访问它们。...基类的private成员在私有派生类中是不可直接访问的,所以无论是派生类成员还是通过派生类的对象,都无法直接访问基类中的private成员。
,这里是Animal指针指向,所以他就调用Animal里面的,普通函数是父类为子类提供的“强制实现”,也就是只要是父类指针调用普通函数,那就是父类的普通函数 而虚函数的作用,主要是为了让父类指针可以调用子类的函数...,这种是在运行时才决定调用哪个函数 1、虚函数: C++的虚函数主要作用是“运行时多态”,父类中提供虚函数的实现,为子类提供默认的函数实现。...子类可以重写父类的虚函数实现子类的特殊化。 2、纯虚函数: C++中包含纯虚函数的类,被称为是“抽象类”。抽象类不能使用new出对象,只有实现了这个纯虚函数的子类才能new出对象。 ...3、普通函数: 普通函数是静态编译的,没有运行时多态,只会根据指针或引用的“字面值”类对象,调用自己的普通函数。 普通函数是父类为子类提供的“强制实现”。 ...因此,在继承关系中,子类不应该重写父类的普通函数,因为函数的调用至于类对象的字面值有关。 参考链接
二、转换的本质 派生类可以转换为基类的本质是: ①为什么派生类可以转换为基类:派生类从基类而来,因此派生类中包含了基类的方法和成员。...此时基类可以通过指针或引用指向派生类(相当于将派生类从基类中继承的那部分方法和成员绑定到基类上了,相当于派生类被截断了),然后基类就可以将派生类假装是一个基类对象来使用(调用其中的成员/方法) ②为什么基类不能转换为派生类...如果将一个基类对象绑定到派生类的指针/引用上,此时派生类通过指针/引用访问自己新定义的成员/方法时,发现找不到(因此不能将基类转换为派生类) 例如:下面B继承于A,子类继承于父类,同时为父类的成员开辟了空间...//假设B公有继承于A A *a; B b; a = &b; //将派生类转换为基类,正确 B *p = a; //将基类再转换为派生类,错误 五、类静态类型/类动态类型 在上面我们介绍过,基类的指针或引用可以指向于基类对象也可以指向于派生类对象...演示案例② 我们修改演示案例①,上面是将基类的指针指向于派生类。
四、派生类到基类的转换 当派生类以public方式继承基类时,编译器可自动执行的转换(向上转型 upcasting 安全转换) 派生类对象指针自动转化为基类对象指针 派生类对象引用自动转化为基类对象引用...派生类对象自动转换为基类对象(特有的成员消失) 当派生类以private/protected方式继承基类时 派生类对象指针(引用)转化为基类对象指针(引用)需用强制类型转化。...将派生类对象看成基类对象 //pm = &e1; // 基类对象指针无法转化为派生类对象指针。...无法将基类对象看成是派生类对象 e1 = m1; // 派生类对象可以转化为基类对象。将派生类对象看成基类对象 // 会产生对象切割(派生类特有成员消失)。...; // 基类对象无法强制转化为派生类对象 return 0; } 五、基类到派生类的转换 基类对象指针(引用)可用强制类型转换为派生类对象指针(引用), 而基类对象无法执行这类转换.
派生类与基类之间的关系: 派生类对象可以使用基类(公有的)方法。 基类指针可以在不进行显示类型转换的情况下指向派生类对象,但只能调用基类方法。...将派生类引用或指针转换为基类引用或指针称为向上强制转换,该转换使得公有继承不需要进行显示类型转换。...且该转换是可以传递的,例如基类A,其派生类AP,AP的派生类APP,则A指针或引用可以指向或引用AP类对象和APP类对象。 相反的,我们将基类指针或引用转换为派生类指针或引用称为向下强制转换。...虚函数表存储了为类对象进行声明的虚函数的地址。通常情况下,基类对象包含一个指向该类中虚函数表的指针。...在派生类对象中将包含一个自己的虚函数表,如果派生类没有重新定义虚函数,则派生类的虚函数列表保函数的原始版本地址;如果派生类提供了虚函数的新定义,则该虚函数列表对应的位置将保存新函数的地址;如果派生类新增加虚函数
,所以派生类中包含基类的那部分成员肯定会包含基函数表的指针,但是派生类的虚函数应该怎么存放呢?...(vTabled); } 结果如下: 我们发现基类对象和派生类对象的虚函数表是不同的,并且派生类对象虚函数表中存放了两个虚函数的地址,其中一个与基类的虚函数地址相同也就是Func1()的地址,另一个则是派生类自己定义的虚函数...综上所述,如果派生类和基类都定义了自己的虚函数,并且基类的虚函数没有在派生类中重写的话,那么派生类中虚函数的地址会存放在派生类继承的基类那部分的虚函数表中的末尾,并且基类定义的对象和派生类定义的对象的虚函数表的地址是不同的...,派生类中的虚函数的地址是存放在继承的基类的虚函数表中的,那么对于重写的虚函数是写在基类虚函数表的末尾,还是将基类被重写的虚函数地址覆盖呢?...只是因为子类会继承基类的虚函数,所以基类的虚函数指针也会存在该子类的虚函数表中,相当于将基类的虚函数表直接继承下来,再将子类自己的虚函数指针存放进去,子类也就不用自己再生成一个虚函数表。
回忆一下,在单继承和多重继承的情况下,内嵌的基类实例地址比起派生类实例地址来,要么地址相同(单继承,以及多重继承的最靠左基类) ,要么地址相差一个固定偏移量(多重继承的非最靠左基类) 。...然而,当虚继承时,一般说来,派生类地址和其虚基类地址之间的偏移量是不固定的,因为如果这个派生类又被进一步继承的话,最终派生类会把共享的虚基类实例数据放到一个与上一层派生类不同的偏移量处。...当使用指针访问虚基类成员变量时,由于指针可以是指向派生类实例的基类指针,所以,编译器不能根据声明的指针类型计算偏移,而必须找到另一种间接的方法,从派生类指针计算虚基类的位置。...这些指针成员变量指向派生类的虚基类,每个虚基类一个指针。这种方式的优点是:获取虚基类地址时,所用代码比较少。然而,编译器优化代码时通常都可以采取措施避免重复计算虚基类地址。...况且,这种实现方式还有一个大弊端:从多个虚基类派生时,类实例将占用更多的内存空间;获取虚基类的虚基类的地址时,需要多次使用指针,从而效率较低等等。
把继承结构,也就是说成从上(基类)到下(派生类)的结构 基类对象 < -派生类对象 类型从下到上的转换(可以) 派生类对象 <- 基类对象 类型从上到下的转换(不可以) 基类指针(引用)<- 派生类对象...(引用)调用同名覆盖方法(虚函数),基类指针指向哪个派生类对象,就会调用哪个派生类对象的同名覆盖方法。...多重继承 代码复用 一个派生类有多个基类 virtual可以修饰继承方式,是虚继承,虚继承的类是虚基类 基类指针指向派生类对象,永远指向的是派生类基类部分数据的起始地址。...当一个类同时继承了两个或更多个共同基类,而这些基类又继承自同一个共同的基类时,就会形成菱形继承结构。为了解决由此可能产生的二义性和数据重复的问题,可以将这些共同的基类声明为虚基类。...通过使用 dynamic_cast 将 basePtr 转换为 Derived* 类型的指针 derivedPtr,我们可以安全地调用 Derived 类的方法。
协变 类虚函数返回基类对象的指针或者引用,派生类虚函数返回派生类对象的指针或者引用时,称为协变(基类与派生类虚函数返回值类型不同) 示例: class A {}; class B : public...virtual,派生类的虚函数加virtual,这种情况是不构成虚函数的 析构函数的重写 我们知道,基类指针和引用可以指向基类和派生对象,由此通过指针和引用释放对象时需要实现析构的多态,但基类与派生类析构函数名字不同...(虚表指针也就),存在部分的另一部分是自己的成员 对于派生类d对象,因为Func1完成了重写,所以d的虚表中存的是重写的Derive::Func1(将继承的虚函数进行重写,而对应的在虚函数表上进行覆盖成自己的虚函数地址也叫作覆盖...对于虚函数类会在对象的成员变量中生成虚函数表指针,指向的虚函数表中储 了该对象的虚函数地址 对于派生类会继承基类的虚函数表,如果派生类重写了虚函数,则会对继承的虚函数表中对应的函数地址进行覆盖成派生类对象的虚函数...可以,并且最好把基类的析构函数定义成虚函数 当基类指针指向new出来的基类对象或new出来的派生类对象时,需要使用指针进行释放对象,此时需要析构函数为虚函数,保证指针指向的对象成功释放 对象访问普通函数快还是虚函数更快
多态的实现 基本条件 通过基类的指针或引用调用虚函数:多态的前提是通过基类的指针或引用来访问派生类对象。...虚函数重写的其他问题 协变 当派生类重写基类的虚函数时,如果基类虚函数返回基类类型的指针或引用,派生类虚函数可以返回派生类类型的指针或引用。这种情况称为协变。...这样就是实现了指针或引用指向基类或者派生类直接调用指向类的虚函数。实现了多态。 动态绑定和静态绑定 静态绑定:编译器在编译时已经确定了函数调用的地址,通常用于普通函数(不满足多态条件)。...当基类中的虚函数未被派生类重写时,派生类的虚表会继承这些地址。 派生类的虚函数表:当派生类重写了基类的虚函数,派生类的虚表中的相应条目会替换为派生类的虚函数地址。...派生类的虚表包含三类地址: 基类的虚函数地址:未被派生类重写的基类虚函数。 重写的虚函数地址:派生类对基类虚函数的重写。 派生类特有的虚函数地址:派生类定义的独有虚函数。
「派生类的指针」可以赋给「基类指针」; 通过基类指针调用基类和派生类中的同名「虚函数」时: 若该指针指向一个基类的对象,那么被调用是 基类的虚函数; 若该指针指向一个派生类的对象,那么被调用 的是派生类的虚函数...— — || 01 虚函数表 每一个有「虚函数」的类(或有虚函数的类的派生类)都有一个「虚函数表」,该类的任何对象中都放着虚函数表的指针。「虚函数表」中列出了该类的「虚函数」地址。...多态的函数调用语句被编译成一系列根据基类指针所指向的(或基类引用所引用的)对象中存放的虚函数表的地址,在虚函数表中查找虚函数地址,并调用虚函数的指令。...解决办法:把基类的析构函数声明为virtual 派生类的析构函数可以 virtual 不进行声明; 通过基类的指针删除派生类对象时,首先调用派生类的析构函数,然后调用基类的析构函数,还是遵循「先构造,后虚构...,则应该将析构函数也定义成虚函数; 或者,一个类打算作为基类使用,也应该将析构函数定义成虚函数。
覆盖虚函数:如果派生类重写了基类的虚函数,则派生类的虚表中会用派生类的函数覆盖基类的函数。 派生类新函数:派生类新增的虚函数会被添加到虚表的末尾。...多态需求强的场景:当需要通过基类指针或引用调用派生类的不同实现时,动态绑定是必不可少的。虚函数通过虚表机制,可以在运行时调用不同的派生类函数。...当派生类重写了基类的虚函数时,虚表中的基类函数指针会被派生类函数的指针替换。如果派生类定义了新的虚函数,这些新的虚函数指针将会追加到派生类的虚表末尾。...派生类可以通过这个虚表调用基类中的虚函数。 重写虚函数:如果派生类重写了基类的虚函数,则派生类的虚表中相应的函数指针将会被覆盖为派生类的函数实现,重写的虚函数替换基类的虚函数。...派生类会为每个基类维护一个独立的虚表,来存储对应基类的虚函数指针。当调用虚函数时,派生类会根据继承自哪个基类,选择相应的虚表来查找虚函数的地址。
虚函数,这样,将不同的派生类对象的地址赋给基类的指针变量后, 就可以动态地根据这种赋值语句调用不同类中的函数。...但 可以定义指向抽象类的指针,即指向这种基类的指针。当用这种基类 指针指向其派生类的对象时,必须在派生类中重载纯虚函数,否则会 产生程序的运行错误。...,即相对地址(距离开始位置的字节数) 如:mptr = &S::y; 表示将数据成员y的相对起始地址赋给指针变量mptr。...在使用这种指向成员函数的指针前,应先对其赋值 PointName= ClassName::FuncName; 同样地,只是将指定成员函数的相对地址赋给指向成员函数的指针。...基类为形状类,派生类为矩形类和三角形类,通过基类指针指向不同的派生类对象,调用求面积函数,得到相应的面积结果,实现多态性。
3、C++标准规定对对象取地址将始终为对应类型的首地址,这样的话如果试图取基类类型的地址,将取到的则是基类部分的首地址。...,这是因为我们将基类的析构函数声明为虚函数的原因,在pI 指向派生类首地址的前提下,如果~IRectangle() 是虚函数,那么会找到实际的函数~Rectangle() 执行,而~Rectangle...如果没有这样做的话,只会输出基类的 析构函数,这种输出情况通过比对规则2也可以理解,pI 现在虽然指向派生类对象首地址,但执行pI->~IRectangle() 时 发现不是虚函数,故直接调用, 假如在派生类析构函数内有释放内存资源的操作...因为此时基类是空类1个字节,派生类有虚函数故有vptr 4个字节,基类“继承”的1个字节附在vptr下面,现在的p 实际上是指向了附属1字节,即operator delete(void*) 传递的指针值已经不是...将基类析构函数改成虚函数,fun() 最好也改成虚函数,只要有一个虚函数,基类大小就为一个vptr ,此时基类和派生类大小都是4个字节,p也指向派生类的首地址,问题解决,参考规则3。
如果基类有3个虚函数,那么基类的虚表中就有三项(虚函数地址),派生类也会有虚表,至少有三项,如果重写了相应的虚函数,那么虚表中的地址就会改变,指向自身的虚函数实现。...如果派生类有自己的虚函数,那么虚表中就会添加该项。派生类的虚表中虚函数地址的排列顺序和基类的虚表中虚函数地址排列顺序相同。...如果派生类重写了基类的虚方法,该派生类虚函数表将保存重写的虚函数的地址,而不是基类的虚函数地址。...如果基类中的虚方法没有在派生类中重写,那么派生类将继承基类中的虚方法,而且派生类中虚函数表将保存基类中未被重写的虚函数的地址。...当存在多重继承时,多重继承了几个基类,子类将含有几个虚指针,并且此指针具有传递性。
运行时多态允许通过基类指针或引用来调用派生类的函数,实现了动态绑定。 2.多态的定义和实现 2.1多态的构成条件 多态的实现通常依赖于虚函数。在基类中声明虚函数,然后在派生类中进行重写(覆盖)。...通过基类指针或引用调用虚函数时,将根据对象的实际类型调用相应的派生类函数 从上面这段话我们知道在继承中要构成多态还有两个条件: 必须通过基类的指针或者引用调用虚函数 被调用的函数必须是虚函数...即基类虚函数返回基类对象的指针或者引用,派生类虚函数返回派生类对象的指针或者引用时,称为协变 class A { public: virtual A* f() { return new...基类指针或引用调用虚函数时,编译器生成的机器代码确实会先访问对象的虚函数指针(vptr),再通过虚函数表(vtable)找到实际要调用的虚函数的地址,最终进行调用。...多态必须使用基类的指针/引用来调用虚函数的原因主要是因为基类指针/引用可以在运行时指向派生类对象,而且能正确地调用派生类的虚函数。
五条基本规则: 1、如果基类已经插入了vptr, 则派生类将继承和重用该vptr。...3、C++标准规定对对象取地址将始终为对应类型的首地址,这样的话如果试图取基类类型的地址,将取到的则是基类部分的首地址。...,这是因为我们将基类的析构函数声明为虚函数的原因,在pI 指向派生类首地址的前提下,如果~IRectangle() 是虚函数,那么会找到实际的函数~Rectangle() 执行,而~Rectangle...如果没有这样做的话,只会输出基类的 析构函数,这种输出情况通过比对规则2也可以理解,pI 现在虽然指向派生类对象首地址,但执行pI->~IRectangle() 时 发现不是虚函数,故直接调用, 假如在派生类析构函数内有释放内存资源的操作...将基类析构函数改成虚函数,fun() 最好也改成虚函数,只要有一个虚函数,基类大小就为一个vptr ,此时基类和派生类大小都是4个字节,p也指向派生类的首地址,问题解决,参考规则3。
领取专属 10元无门槛券
手把手带您无忧上云