C++中派生类对基类成员的访问形式主要有以下两种: 1、内部访问:由派生类中新增成员对基类继承来的成员的访问。 2、对象访问:在派生类外部,通过派生类的对象对从基类继承来的成员的访问。...今天给大家介绍在3中继承方式下,派生类对基类成员的访问规则。...基类的private成员在私有派生类中是不可直接访问的,所以无论是派生类的成员还是通过派生类的对象,都无法直接访问从基类继承来的private成员,但是可以通过基类提供的public成员函数间接访问。...基类的private成员在私有派生类中是不可直接访问的,所以无论是派生类成员还是派生类的对象,都无法直接访问从基类继承来的private成员,但是可以通过基类提供的public成员函数直接访问它们。...基类的private成员在私有派生类中是不可直接访问的,所以无论是派生类成员还是通过派生类的对象,都无法直接访问基类中的private成员。
二、转换的本质 派生类可以转换为基类的本质是: ①为什么派生类可以转换为基类:派生类从基类而来,因此派生类中包含了基类的方法和成员。...此时基类可以通过指针或引用指向派生类(相当于将派生类从基类中继承的那部分方法和成员绑定到基类上了,相当于派生类被截断了),然后基类就可以将派生类假装是一个基类对象来使用(调用其中的成员/方法) ②为什么基类不能转换为派生类...如果将一个基类对象绑定到派生类的指针/引用上,此时派生类通过指针/引用访问自己新定义的成员/方法时,发现找不到(因此不能将基类转换为派生类) 例如:下面B继承于A,子类继承于父类,同时为父类的成员开辟了空间...//假设B公有继承于A A *a; B b; a = &b; //将派生类转换为基类,正确 B *p = a; //将基类再转换为派生类,错误 五、类静态类型/类动态类型 在上面我们介绍过,基类的指针或引用可以指向于基类对象也可以指向于派生类对象...B b(20); //定义B类对象 A a2(a1); //拷贝构造,使用与A类类型a1对象 A a3(b); //拷贝构造,使用B类类型的b对象,b对象的内容被截断 return 0; } 赋值运算符时的类型转换
一、不能自动继承的成员函数 构造函数(包括拷贝构造函数) 析构函数 =运算符 二、继承与构造函数 基类的构造函数不被继承,派生类中需要声明自己的构造函数。...从输出可以看出: 派生类对象的构造次序: 先调用基类对象成员的构造函数,接着是基类的构造函数,然后是派生类的对象成员的构造函数,最后是派生类自身的构造函数。...:xx 访问,如b.b_ 访问存在歧义,实际上static成员不属于任一对象。...四、派生类到基类的转换 当派生类以public方式继承基类时,编译器可自动执行的转换(向上转型 upcasting 安全转换) 派生类对象指针自动转化为基类对象指针 派生类对象引用自动转化为基类对象引用...; // 基类对象无法强制转化为派生类对象 return 0; } 五、基类到派生类的转换 基类对象指针(引用)可用强制类型转换为派生类对象指针(引用), 而基类对象无法执行这类转换.
.152: Never assign a pointer to an array of derived class objects to a pointer to its base C.152:永远不要将派生类数组的指针赋值给基类指针...作为赋值结果的基类指针的下标运算会引起无效的对象访问并可能发生内存破坏。...提示所有数组退化和基类类型向派生类类型转换的情况。...suffer a derived-to-base conversion before getting into the span 使用span传递数组而不是指针,也不要再放入span之前让数组名经过一次派生类向基类类型的转换
C.87: Beware of == on base classes C.87:小心基类的相等运算符 Reason(原因) It is really hard to write a foolproof...为继承体系写出简单又好用的相等运算符真的很难。...B的相等比较运算符的第二个操作数接受类型转换,但是第一个不行。...,但是简单的方法不行。...提示被定义为虚函数的相等比较运算符;其他比较运算符也一样:!=, , 和 >=。
---- C++中函数的形参列表里可以有占位参数,用来做占位,调用函数时必须填补该位置 语法: 返回值类型 函数名 (数据类型){} 在现阶段函数的占位参数存在意义不大,但是后面的课程中会用到该技术.../*所以基类的指针操作派生类的对象时, 由于基类指针会向操作基类对象那样操作派生类对象, 而基类对象所占用的内存空间通常小于派生类对象, 所以基类指针不会超出派生类对象去操作数据...,当创建派生类对象,也会调用基类的构造函数 问题:基类和派生类的构造和析构顺序是谁先谁后?...: 派生类对象可以直接访问到派生类中同名成员 派生类对象加作用域可以访问到基类同名成员 当派生类与基类拥有同名的成员函数,派生类会隐藏基类中同名成员函数,加作用域可以访问到基类中同名函数 ---- 4.6.7...--- C++允许一个类继承多个类 语法: class 派生类 :继承方式 基类1 , 继承方式 基类2...
基类 & 派生类 一个类可以派生自多个类,这意味着,它可以从多个基类继承数据和函数。定义一个派生类,我们使用一个类派生列表来指定基类。...因此基类成员如果不想被派生类的成员函数访问,则应在基类中声明为 private。...当使用不同类型的继承时,遵循以下几个规则: 公有继承(public):当一个类派生自公有基类时,基类的公有成员也是派生类的公有成员,基类的保护成员也是派生类的保护成员,基类的私有成员不能直接被派生类访问...C++ 类可以从多个类继承成员,语法如下: class :,,… { }; 其中,访问修饰符继承方式是 public、protected...是函数所使用的数据类型的占位符名称。
这就是多态:将派生类对象视为基类对象,并执行派生类的Swim()实现。...抽象基类和纯虚函数 不能实例化的基类被称为抽象基类,这样的基类只有一个用途,那就是从它派生出其他类。在 C++中,要创建抽象基类,可声明纯虚函数。...这让基类可指定派生类中方法的名称和特征(Signature),即指定派生类的接口。虽然不能实例化抽象基类,但可将指针或引用的类型指定为抽象基类。...如果派生类可能被用作基类,派生它是最好使用关键字virtual: class Derived1 : public virtual Base { //members and funnctions...注意:C++关键字virtual的含义随上下文而异(我想这样做的目的很可能是为了省事),对其含义总结如下: 在函数声明中,virtual意味着当基类指针指向派生对象时,通过它可调用派生类的相应函数。
基类 & 派生类 一个类可以派生自多个类,这意味着,它可以从多个基类继承数据和函数。定义一个派生类,我们使用一个类派生列表来指定基类。...因此基类成员如果不想被派生类的成员函数访问,则应在基类中声明为 private。...当使用不同类型的继承时,遵循以下几个规则: 公有继承(public):当一个类派生自公有基类时,基类的公有成员也是派生类的公有成员,基类的保护成员也是派生类的保护成员,基类的私有成员不能直接被派生类访问...C++ 类可以从多个类继承成员,语法如下: class :,,…{}; 其中,访问修饰符继承方式是 public、protected...是函数所使用的数据类型的占位符名称。
为了在派生类中使用与基类同名的成员,必须在该成员名之前加上基类名和作用域标识符“::”,即 基类名::成员名 访问声明 访问声明的方法就是把基类的保护成员或共有成员直接写在私有派生类定义式中的同名段中,...若虚基类由非虚基类派生而来,则仍然先调用基类构造函数,再调用派生类的构造函数。 ~ 5.6 赋值兼容规则 在一定条件下,不同类型的数据之间可以进行类型转换,如可以将整型数据赋值给双精度型变量。...~ 6.3 虚析构函数 如果在主函数中用new运算符建立一个派生类的无名对象和定义一个基类的对象指针,并将无名对象的地址赋值给这个对象指针,当用delete运算符撤销无名对象时,系统只执行基类的析构函数...如果希望程序执行动态连编方式,在用delete运算符撤销派生类的无名对象时,先调用派生类的析构函数,再调用基类的析构函数,可以将基类的析构函数声明为虚析构函数。...类型参数(通常用C++标识符表示,如T、type等)实际上是一个虚拟的类型名,使用前并未指定它是哪一种具体的类型,但使用函数模板时,必须将类型实例化。
只有成员函数中的代码才应该使用作用域操作符覆盖虚函数机制。 为什么会希望覆盖虚函数机制?最常见的理由是为了派生类虚函数调用基类中的版本。...派生类虚函数调用基类版本时,必须显式使用作用域操作符。如果派生类函数忽略了这样做,则函数调用会在运行时确定并且将是一个自身调用,从而导致无穷递归。...与基类成员同名的派生类成员将屏蔽对基类成员的直接访问。...使用作用域操作符访问被屏蔽成员 可以使用作用域操作符访问被屏蔽的基类成员: struct Derived : Base { int get_base_mem() { return Base::...“隐藏”是指派生类的函数屏蔽了与其同名的基类函数,规则如下: (1)如果派生类的函数与基类的函数同名,但是参数不同。
我们可以在派生类方法当中使用它,但无法通过派生类对象直接调用,但无法访问基类的私有方法和对象。 这个概念我们很好理解,但具体到实现上,我们如何在派生类的方法当中调用基类的公有或者保护方法呢?...由于Student类是继承了valarray类,而非包含一个valarray的实例,那么我们如何在Average方法的实现当中调用valarray的公有方法呢? 答案是通过解析运算符和类名调用。...现在我们知道了使用解析运算符可以访问基类的方法,但问题又来了,如果我们要访问的不再是基类的方法而是基类对象本身,那么我们应该怎么办呢?...访问基类的友元 方法和对象都好办,但友元怎么办呢?因为友元函数并不属于类,所以无法通过类名或者解析运算符搞定。针对这种情况,我们只能取巧,通过显式地对派生类进行类型转换实现。...原因也简单,首先在不使用类型转换的情况下,os << stu和自身的友元函数类型匹配, 将会发生递归。其次由于C++支持多继承,所以编译器将会无法确定转换成哪个基类的类型。
文章目录 前言 一、面向对象 二、使用类 1.类的基础使用 2.运算符重载 3.友元 1.友元函数 2.友元类 3.友元成员函数 三、类继承 基类 派生类 插个访问权限的事情 2.多态 ***公有 **...2.运算符重载 首先我们明确一点,运算符重载是c++多态的一种形式,我们来类比与函数重载来看:同名的函数来完成相同的基本操作,即使被利用与不同的数据类型,运算符重载也差不多 例如:运算符“ * ”一样在指针中用来解引用...虚函数 虚函数源于c++中的类继承,是多态的一种。在c++中,一个基类的指针或者引用可以指向或者引用派生类的对象。同时,派生类可以重写基类中的成员函数。...基类中 可以在基类中将被重写的成员函数设置为虚函数,其含义是:当通过基类的指针或者引用调用该成员函数时,将根据指针指向的对象类型确定调用的函数,而非指针的类型。...编译器要求您创建具体类(如 Tuna)的对象,这与现实世界一致。
想要从根源上解决问题,需要使用C++官方提供的一个新的功能——虚基类。 虚基类可以使得从多个类派生出的对象只继承一个基类对象,要使用虚基类,需要在类声明当中使用关键字virtual。...虽然这里使用的也是virtual关键字,但虚函数和虚基类之间并没有任何关联,只不过是C++官方不愿意引入更多关键字以免造成使用者的负担而已。...使用了虚基类之后,一些语法会和之前有所不同,接下来我们来详细介绍。 构造函数 对于非虚基类的继承关系来说,我们可以在构造函数当中将数据传递给基类。...为了避免这种冲突,C++在基类是虚时,禁止数据通过中间类传递给虚基类,因此上述代码中的wk参数将会失效。在这种情况下,编译器会调用Worker的默认构造函数。...怎么解决呢,一个方法是使用作用域解析符: sw.Singer::show(); 但显然这种做法非常麻烦,更好一点的做法是在SingingWaiter中重新定义show函数,在函数内部指出要调用哪个show
: 继承关系和访问限定符: 继承基类成员的访问方式变化 类成员/继承方式 public继承 protected继承 private继承 基类的public成员 派生类的public成员 派生类的protected...; } protected: int _num = 999; // 学号 }; void Test() { Student s1; s1.Print(); }; 这段代码展示了成员隐藏,以及如何在派生类中访问基类的被隐藏成员的概念...如果在派生类中尝试访问一个被隐藏的基类成员,需要显式地使用类名限定符来指定基类的成员。在 Print 方法中使用 Person::_num 来访问基类 Person 中的 _num 成员。...C++ 规则规定,如果派生类提供了和基类同名的函数,基类中同名的函数在派生类的作用域就不再可见了 因此,在 B 类的成员函数 fun(int) 中,调用 fun() 试图无参数调用被隐藏的同名函数会无法编译...,因为编译器认为我们试图调用 fun(int) 这个版本,但没有提供参数,导致参数不匹配 修复 为了调用基类 A 的 fun 函数,我们必须显式地使用作用域解析运算符 :: 来指明我们想要调用的函数属于基类作用域
图5-6 动态联编示例 5.4 虚函数 虚函数是动态束定的基础 虚函数是非static成员函数 格式:virtual () 若类中一成员函数被说明为虚函数,则该成员函数在派生类中可能有不同的实现...act1() 5.5 纯虚函数 在基类中不能为虚函数给出一个有意义的实现时,可将其声明为纯虚函数,其实现留待派生类完成 5.5.1 纯虚函数定义格式 ?...图5-9 纯虚函数 5.6 抽象类 带有纯虚函数的类称为抽象类 抽象类只能作为基类使用,其纯虚函数的实现由派生类给出;但派生类仍可不给出纯虚函数的定义,继续作为抽象类存在 抽象类不能定义对象,一般将该类的构造函数说明为保护的访问控制权限...在成员函数内可以调用纯虚函数,但在构造函数或析构函数内不能调用纯虚函数(纯虚函数没有实现代码) 5.6.1 抽象类的作用 1用作基类:在一个继承层次结构中,提供一个公共的根,并基于抽象类的操作设计出对抽象类所描述的一类对象进行操作的公共接口...,其完整的实现由派生类完成 2用作指针或引用的基类型:保证进入继承层次的每个类都具有(提供)纯虚函数所要求的行为 ?
基类的公有成员在派生类中仍然是公有的 基类的保护成员在派生类中仍然是受保护的 基类的不可访问的成员在派生类中仍然是不可访问的 基类的私有成员在派生类中变成了不可访问的 总结:在公有派生的情况下,通过派生类自己的成员函数可以访问继承过来的公有和保护成员...01 作用域分辨符和成员名限定 代码格式: 类名::标识符 :: 为作用域分辨符,"类名"可以是任一基类或派生类名,“标识符”是该类中声明的任一成员名, 像这样: // 基类A, 也叫根类 class...如果我们想要访问被隐藏的基类中的成员则使用作用域分辨符B::gunc(); A func B func */ } 02 派生类支配基类的同名函数 如果派生类定义了一个同基类成员函数同名的新成员函数...在这里,直接使用成员名只能访问派生类中的成员函数,使用作用域运算符,才能访问基类的同名成员函数。 派生类中的成员函数名支配基类中的同名的成员函数名,这称为名字支配规则。...例如上个例子中 c.gunc() // 输出”C gunc”, 基类B中的gunc成员函数被支配了 c.B::gunc(); // 加上作用域分辨符, 来使用被支配的成员 总结 C++中的多重继承可能更灵活
领取专属 10元无门槛券
手把手带您无忧上云