1.什么是基类? 在面向对象设计中,被定义为包含所有实体共性的class类型,被称为“基类”。-百度百科 简单理解,即父类(相对派生类) 2.什么是派生类? ...利用继承机制,新的类可以从已有的类中派生。那些用于派生的类称为这些特别派生出的类的“基类”。 简单理解,即子类(相对基类) 3.两者关联 基类和派生类是一个相对的关系。...基类和派生类反映了类与类的继承关系,是相对而言的。基类又称父类,是被派生类继承的类。派生类又称子类,是从一个已有类的基础上创建的新类,新类包含基类的所有成员,并且还添加了自己的成员。...4.实际例子 假设有两个类A和B,A和B都需要实现一个打印的功能,原始的做法是A写一个打印函数,B也写一个打印函数。两个类还好可以写,但多了就特别麻烦。...这个时候我们就可以写一个类C,C里面写一个打印函数。A和B分别继承C,这样A和B就不要写打印函数了。这样即节省了代码,又优化了结构。 上面的情况,C是A和B的基类,A和B是C的派生类。
基类指针指向派生类,我们已经很熟了。假如我们想用派生类反过来指向基类,就需要有两个要求:1)马克-to-win:基类指针开始时指向派生类,2)我们还需要清清楚楚的转型一下。
通常在层次关系的根部有一个基类,其他类则直接或间接的从基类继承而来,这些继承得到的类称为派生类。基类负责定义在层次关系中所有类共同拥有的成员,而每个派生类定义各自特有的成员。...成员函数与继承派生类可以继承其基类的成员, 然而有时候派生类必须对其重新定义。派生类需要对这些操作提供自己的新定义以覆盖从基类继承而来的旧定义。...派生类可以继承定义在基类中的成员,但是派生类的成员函数不一定有权访问从基类继承而来的成员。 和其他使用基类的代码一样,派生类能访问公有成员,而不能访问私有成员。...派生类必须通过使用类派生列表(clss erivatin list)明确指出它是从哪个(哪些)基类继承而来的。...现在,我们只需知道访问说明符的作用是控制派生类从基类继承而来的成员是否对派生类的用户可见。 如果一个派生是公有的,则基类的公有成员也是派生类接口的组成部分。
覆盖(Override)是指派生类中存在重新定义的函数,其函数名、参数列、返回值类型必须同父类中的相对应被覆盖的函数严格一致,覆盖函数和被覆盖函数只有函数体 (花括号中的部分)不同,当派生类对象调用子类中该同名函数时会自动调用子类中的覆盖版本...下面我们从成员函数的角度来讲述重载和覆盖的区别。 成员函数被重载的特征有: 相同的范围(在同一个类中);2) 函数名字相同;3) 参数不同;4) virtual关键字可有可无。...覆盖的特征有: 不同的范围(分别位于派生类与基类);2) 函数名字相同;3) 参数相同;4) 基类函数必须有virtual关键字。...隐藏是指派生类的函数屏蔽了与其同名的基类函数,规则如下: 如果派生类的函数与基类的函数同名,但是参数不同。此时,不论有无virtual关键字,基类的函数将被隐藏(注意别与重载混淆)。...2) 如果派生类的函数与基类的函数同名,并且参数也相同,但是基类函数没有virtual关键字。此时,基类的函数被隐藏(注意别与覆盖混淆)。
参考链接: C++继承 继承 类和类的关系有组合、继承和代理。继承的本质就是代码复用。子类继承父类中的一些东西,父类也称为基类,子类也称为派生类。派生类继承了基类除构造函数以外的所有成员。 ...基类中不同访问限定符下(public、protected、private)的成员以不同的继承方式继承,在派生类中的访问限定也不同,具体如下: 基类的布局优先于派生类 #include<iostream...1.调用基类的构造函数 2.调用派生类的构造函数 派生类的析构可想而知: 1.调用派生类的析构函数 2.调用基类的析构函数 虚函数 如下程序: class Base { public: Base...基类中含有虚函数,那么基类布局中存在一个虚函数指针,指向虚函数表;且其派生类中与其同名同参的函数不需要加virtual也是虚函数。...此时基类和派生类的布局如下: vfptr的指针大小为4(32位机器)。因此基类字节数为8,派生类为12。
.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.ReadKey(); } } } } 结果: 成本:1732.5 C# 多重继承 多重继承指的是一个类别可以同时从多于一个父类继承行为与特征的功能
C++中派生类对基类成员的访问形式主要有以下两种: 1、内部访问:由派生类中新增成员对基类继承来的成员的访问。 2、对象访问:在派生类外部,通过派生类的对象对从基类继承来的成员的访问。...今天给大家介绍在3中继承方式下,派生类对基类成员的访问规则。...基类的private成员在私有派生类中是不可直接访问的,所以无论是派生类的成员还是通过派生类的对象,都无法直接访问从基类继承来的private成员,但是可以通过基类提供的public成员函数间接访问。...基类的private成员在私有派生类中是不可直接访问的,所以无论是派生类成员还是派生类的对象,都无法直接访问从基类继承来的private成员,但是可以通过基类提供的public成员函数直接访问它们。...基类的private成员在私有派生类中是不可直接访问的,所以无论是派生类成员还是通过派生类的对象,都无法直接访问基类中的private成员。
继承,因为protetced/private继承下来的成员都只能在派生类的类里面使用,实际中扩展维护性不强 2.基类和派生类对象赋值转换 派生类对象可以赋值给基类的对象 / 基类的指针 / 基类的引用...在行 Person* pp = &sobj; 中,pp 是指向 Person 类型的指针,但它实际上指向了派生类 Student 的对象 sobj,没有发生切片,因为指针指向的是完整的 Student...无法访问,因为_No是Student特有的成员,即使它实际上存在于sobj中 即使我们通过基类引用或指针操作对象,派生类对象的完整信息(所有成员变量和函数)仍然都在内存中,没有丢失。...使用引用和指针时不会发生切片 对象切片的问题仅在派生类对象被赋值给另一个基类类型的对象时才会发生,比如当派生类对象被传值给一个基类对象的函数参数,或者通过赋值构造一个新的基类对象。...在使用引用或指针时,这种情况并不会发生 基类对象不能赋值给派生类对象 基类的指针或者引用可以通过强制类型转换赋值给派生类的指针或者引用。但是必须是基类的指针是指向派生类对象时才是安全的。
base2 Bird * a1 = new Bird(); return 0; } 为什么输出为base3,因为eye是个普通函数,在编译阶段就确定好是被谁调用,所以他只认哪个指针指向自己...,这里是Animal指针指向,所以他就调用Animal里面的,普通函数是父类为子类提供的“强制实现”,也就是只要是父类指针调用普通函数,那就是父类的普通函数 而虚函数的作用,主要是为了让父类指针可以调用子类的函数...子类可以重写父类的虚函数实现子类的特殊化。 2、纯虚函数: C++中包含纯虚函数的类,被称为是“抽象类”。抽象类不能使用new出对象,只有实现了这个纯虚函数的子类才能new出对象。 ...3、普通函数: 普通函数是静态编译的,没有运行时多态,只会根据指针或引用的“字面值”类对象,调用自己的普通函数。 普通函数是父类为子类提供的“强制实现”。 ...因此,在继承关系中,子类不应该重写父类的普通函数,因为函数的调用至于类对象的字面值有关。 参考链接
二、转换的本质 派生类可以转换为基类的本质是: ①为什么派生类可以转换为基类:派生类从基类而来,因此派生类中包含了基类的方法和成员。...此时基类可以通过指针或引用指向派生类(相当于将派生类从基类中继承的那部分方法和成员绑定到基类上了,相当于派生类被截断了),然后基类就可以将派生类假装是一个基类对象来使用(调用其中的成员/方法) ②为什么基类不能转换为派生类...如果将一个基类对象绑定到派生类的指针/引用上,此时派生类通过指针/引用访问自己新定义的成员/方法时,发现找不到(因此不能将基类转换为派生类) 例如:下面B继承于A,子类继承于父类,同时为父类的成员开辟了空间...(或指针)所绑定的对象的真实类型,该对象可能是基类的对象,也可能是派生类的对象。...演示案例② 我们修改演示案例①,上面是将基类的指针指向于派生类。
从输出可以看出: 派生类对象的构造次序: 先调用基类对象成员的构造函数,接着是基类的构造函数,然后是派生类的对象成员的构造函数,最后是派生类自身的构造函数。...四、派生类到基类的转换 当派生类以public方式继承基类时,编译器可自动执行的转换(向上转型 upcasting 安全转换) 派生类对象指针自动转化为基类对象指针 派生类对象引用自动转化为基类对象引用...派生类对象自动转换为基类对象(特有的成员消失) 当派生类以private/protected方式继承基类时 派生类对象指针(引用)转化为基类对象指针(引用)需用强制类型转化。...将派生类对象看成基类对象 //pm = &e1; // 基类对象指针无法转化为派生类对象指针。...; // 基类对象无法强制转化为派生类对象 return 0; } 五、基类到派生类的转换 基类对象指针(引用)可用强制类型转换为派生类对象指针(引用), 而基类对象无法执行这类转换.
,取基类指针一定是偏移的。....从而就导致了基类的析构函数被调用了,而派生类的析构函数没有调用这个问题发生....Draw(int) 的虚函数,于是查找基类是否定义了Draw(int),还是没有,就出错了,从出错提示也可以看出来:“IRectangle::Draw”: 函数不接受 1 个参数。...因为此时基类是空类1个字节,派生类有虚函数故有vptr 4个字节,基类“继承”的1个字节附在vptr下面,现在的p 实际上是指向了附属1字节,即operator delete(void*) 传递的指针值已经不是...将基类析构函数改成虚函数,fun() 最好也改成虚函数,只要有一个虚函数,基类大小就为一个vptr ,此时基类和派生类大小都是4个字节,p也指向派生类的首地址,问题解决,参考规则3。
五条基本规则: 1、如果基类已经插入了vptr, 则派生类将继承和重用该vptr。...,取基类指针一定是偏移的。...Draw(int) 的虚函数,于是查找基类是否定义了Draw(int),还是没有,就出错了,从出错提示也可以看出来:“IRectangle::Draw”: 函数不接受 1 个参数。...因为此时基类是空类1个字节,派生类有虚函数故有vptr 4个字节,基类“继承”的1个字节附在vptr下面,现在的p 实际上是指向了附属1字节,即operator delete(void*) 传递的指针值已经不是...将基类析构函数改成虚函数,fun() 最好也改成虚函数,只要有一个虚函数,基类大小就为一个vptr ,此时基类和派生类大小都是4个字节,p也指向派生类的首地址,问题解决,参考规则3。
这就是多态:将派生类对象视为基类对象,并执行派生类的Swim()实现。...为什么需要虚构函数 上面的代码如果加入析构函数释放内存,对于使用new在自由储存区中实例化的派生类对象,如果将其赋值给基类指针,并通过该指针调用delete,将不会调用派生类的析构函数,这可能导致资源未释放...抽象基类和纯虚函数 不能实例化的基类被称为抽象基类,这样的基类只有一个用途,那就是从它派生出其他类。在 C++中,要创建抽象基类,可声明纯虚函数。...这让基类可指定派生类中方法的名称和特征(Signature),即指定派生类的接口。虽然不能实例化抽象基类,但可将指针或引用的类型指定为抽象基类。...注意:C++关键字virtual的含义随上下文而异(我想这样做的目的很可能是为了省事),对其含义总结如下: 在函数声明中,virtual意味着当基类指针指向派生对象时,通过它可调用派生类的相应函数。
如果你真的需要复制语义,就进行深拷贝:提供一个虚的克隆函数,这个函数可以复制实际的派生类型并返回一个指向新对象的所有权指针,同时在派生类中返回派生类型(使用共变量返回类型) 切片问题(slicing...problerm):由派生类实例向基类实例赋值时发生的信息丢失。...共变量返回类型(covariant return type):当基类的虚函数被派生类覆盖时,如果基类的虚函数返回某个类,而派生类返回该类的派生类,也看做是成功的覆盖。...一般情况下,推荐使用智能指针表现所有权(参见R.20)。...但是因为语言规则,共变量返回类型不能是智能指针:当B::clone返回unique_ptr时,D::clone不能返回unique_ptr。
分析上述例程: 派生类对象可以赋值给基类对象(这里对象是广义称法,代指对象、指针、引用),例程中使用基类引用或指针指向派生类对象 如果基类中的 Swim() 不是虚函数,那么无论基类引用(或指针)指向何种类型的对象...如果不将析构函数声明为虚函数,那么如果一个函数的形参是基类指针,实参是指向堆内存的派生类指针时,函数返回时作为实参的派生类指针将被当作基类指针进行析构,这会导致资源释放不完全和内存泄漏;要避免这一问题,...可将基类的析构函数声明为虚函数,那么函数返回时,作为实参的派生类指针就会被当作派生类指针进行析构。...如果派生类没有虚函数,那么派生类虚函数表中的每个元素都指向基类的的虚函数。 派生类对象只含一份 VFT 指针,基类的私有成员都会在派生类对象中占用内存,但基类的 VFT 指针不会在派生类中占用内存。...可见使用虚继承可以解决多继承时的菱形问题,确保 在继承层次结构中,继承多个从同一个类派生而来的基类时,如果这些基类没有采用虚继承,将导致二义性。
指针对象可以使用多态的特性,基类的指针可以指向派生链的任意一个派生类。 指针对象,需要用它的时候,才需要去实例化它,但是在不使用的时候,需要手动回收指针对象的资源。...{ nPublic = 1; // OK nProtected = 1; // error nPrivate =1; // ok,访问从基类继承的...b = d; 派生类对象可以初始化基类引用 Base & br = d; 派生类对象的地址可以赋值给基类指针 Base * pb = & d; 注意:如果派生方式是 private 或 protected...所以派生方式是 private 或 protected,则是无法像 public 派生承方式一样把派生类对象赋值、引用、指针给基类对象。...---- || 03 基类与派生类的指针强制转换 public 派生方式的情况下,派生类对象的指针可以直接赋值给基类指针: Base *ptrBase = & objDerived; ptrBase 指向的是一个
继承 在层次关系根据的类被称为基类bsae class,其他类则直接或间接地从基类继承而来,这些继承得到的类称为派生类derived class。...成员函数如果没被声明成虚函数,那么其解析过程发生在编译时而不是运行时 派生类可以继承定义在基类中的成员,但是派生类的成员函数不一定有权访问从基类继承而来的成员(派生类可以访问公有成员,但是不能访问私有成员...有一点需要注意的是,即使一个基类指针或者引用绑定在一个派生类对象上,我们也不能执行从基类到派生类的转换: Bulk_quote bulk; Quote *itemP = &bulk; //...3.4 存在继承关系的类型之间的转换规则 要想理解在具有继承关系的类之间发生的类型转换,有三点特别重要: 从派生类向基类的类型转换只对指针或者引用类型有效 基类向派生类不存在隐式类型转换 和任何其他成员一样...虚函数与作用域 从名字查找先于类型检查我们可以得知为什么基类和派生类中的虚函数为什么必须有相同的形参列表了。
派生类与基类之间的关系: 派生类对象可以使用基类(公有的)方法。 基类指针可以在不进行显示类型转换的情况下指向派生类对象,但只能调用基类方法。...如果基类中对某个成员函数声明了虚函数,则其派生类中的该成员函数不需要再声明。 从代码维护的层面考虑,随着类的层级的扩展,动态联编提高了代码的灵活性和问题的抽象性,使得程序的维护成本大大降低。...将派生类引用或指针转换为基类引用或指针称为向上强制转换,该转换使得公有继承不需要进行显示类型转换。...且该转换是可以传递的,例如基类A,其派生类AP,AP的派生类APP,则A指针或引用可以指向或引用AP类对象和APP类对象。 相反的,我们将基类指针或引用转换为派生类指针或引用称为向下强制转换。...因此,如果要重新定义继承的方法,则应确保与原来的原型完全相同,但是如果返回类型是基类引用或指针,则可以修改为指向派生类的引用或指针,即允许返回类型随类类型的变化而变化,这种特性被称为返回类型协变。
领取专属 10元无门槛券
手把手带您无忧上云