【c++】继承学习(一):继承机制与基类派生类转换

2024-05-03 21:44

本文主要是介绍【c++】继承学习(一):继承机制与基类派生类转换,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Alt

🔥个人主页Quitecoder

🔥专栏c++笔记仓

Alt

朋友们大家好,本篇文章我们来学习继承部分

目录

  • `1.继承的概念和定义`
    • `继承的定义`
    • `继承基类成员的访问方式变化`
  • `2.基类和派生类对象赋值转换`
  • `3.继承中的作用域`

1.继承的概念和定义

继承(inheritance)机制是面向对象程序设计使代码可以复用的最重要的手段,它允许程序员在保持原有类特性的基础上进行扩展,增加功能,这样产生新的类,称派生类。继承呈现了面向对象程序设计的层次结构,体现了由简单到复杂的认知过程。以前我们接触的复用都是函数复用,继承是类设计层次的复用

通过继承,子类可以重用父类的代码,这有助于减少代码冗余和复杂性,并增加代码的可复用性

子类和父类是继承关系中的两个基本概念:

  1. 父类/ 基类:
    父类是一个更一般的类,它定义了一种通用的数据类型和方法,这些可以被其他类继承。它是继承关系中处于较高层次的类,其特性(属性和方法)可以传递到派生的类中。其他从父类继承的类会自动获得父类定义的所有公共和受保护的成员。

  2. 子类/ 派生类:
    子类是从一个或多个父类继承特性的类。它是继承关系中处于较低层次的类,可以继承其一或多个父类的属性和方法。子类通常会添加一些特有的属性和方法,或者重写某些从父类继承的方法来改变行为。子类集成了父类的特征,并可以拥有自己的特征。

简单来说,父类是派生过程的起点,提供了基础的属性和方法,而子类是继承的结果,它可以扩展和定制继承来的属性和方法。通过这种方式,子类和父类形成了一种层次结构,允许更高层次的代码重用和泛化

例如下面的例子:

在这里插入图片描述

父类包含一些通用的属性,人名和年龄,派生类继承自父类但具有不同的额外特性或方法

class Person
{
public:void Print(){cout << "name:" << _name << endl;cout << "age:" << _age << endl;}
protected:string _name = "jason"; // 姓名int _age = 18;  // 年龄
};
class Student : public Person
{
protected:int _stuid; // 学号
};class Teacher : public Person
{
protected:int _jobid; // 工号
};

继承后父类的Person的成员(成员函数+成员变量)都会变成子类的一部分。这里体现出了Student和Teacher复用了Person的成员

下面我们使用监视窗口查看Student和Teacher对象,可以看到变量的复用。调用Print可以看到成员函数的复用

int main()
{Student s;Teacher t;s.Print();t.Print();return 0;
}

在这里插入图片描述
在这里插入图片描述

继承的定义

格式

在这里插入图片描述
继承关系和访问限定符:

在这里插入图片描述

继承基类成员的访问方式变化

类成员/继承方式public继承protected继承private继承
基类的public成员派生类的public成员派生类的protected成员派生类的private成员
基类的protected成员派生类的protected成员派生类的protected成员派生类的private成员
基类的private成员在派生类中不可见在派生类中不可见在派生类中不可见
  1. 基类private成员在派生类中无论以什么方式继承都是不可见的。这里的不可见是指基类的私有成员还是被继承到了派生类对象中,但是语法上限制派生类对象不管在类里面还是类外面都不能去访问它

我们前面知道,类里面可以访问它的成员,但是private继承下,子类是无法访问父类的成员的

class Person
{
public:void Print(){cout << "name:" << _name << endl;cout << "age:" << _age << endl;}
protected:string _name = "jason"; // 姓名
private:int _age = 18;  // 年龄
};

我们这个类,拥有三个成员

class Student : public Person
{Student(){_name = "peter";}
protected:int _stuid; // 学号
};

在我们这个子类中,我们可以访问除了父类私有成员的其他成员父类的私有成员父类自己可以用,子类不可以直接使用

但是可以间接使用,比如我用子类来调用上面的Print函数

class Student : public Person
{void Fun(){_name = "abc";Print();}
protected:int _stuid; // 学号
};

在这里插入图片描述

  1. 基类private成员在派生类中是不能被访问,如果基类成员不想在类外直接被访问,但需要在派生类中能访问,就定义为protected可以看出保护成员限定符是因继承才出现的

  2. 实际上面的表格我们进行一下总结会发现,基类的私有成员在子类都是不可见。基类的其他成员在子类的访问方式 == 权限小的那个(成员在基类的访问限定符,继承方式),public > protected > private。

  3. 使用关键字class时默认的继承方式是private,使用struct时默认的继承方式是public,不过最好显示的写出继承方式

class Student : protected Person
{
public:void Fun(){_name = "abc";Print();}
protected:int _stuid; // 学号
};

公有的Print函数遇到protected继承变成保护类,无法外部直接调用:

在这里插入图片描述
保护是类外面不能访问,类里面还可以访问

在这里插入图片描述

在实际运用中一般使用都是public继承,几乎很少使用protetced/private继承,也不提倡使用protetced/private继承,因为protetced/private继承下来的成员都只能在派生类的类里面使用,实际中扩展维护性不强

2.基类和派生类对象赋值转换

  1. 派生类对象可以赋值给基类的对象 / 基类的指针 / 基类的引用。这里有个形象的说法叫切片或者切割。寓意把派生类中父类那部分切来赋值过去
class Person
{
protected:string _name; // 姓名string _sex;// 性别int _age; // 年龄
};
class Student : public Person
{
public:int _No; // 学号
};
Student sobj;
// 1.子类对象可以赋值给父类对象/指针/引用
Person pobj = sobj;
Person* pp = &sobj;
Person& rp = sobj;

每一个子类对象都是一个特殊的父类对象
在这里插入图片描述

当派生类对象被赋值给基类对象时会发生。在切片过程中,派生类对象的部分(通常是额外添加的成员变量和方法)会被忽略,只有基类中定义的部分会被复制到基类对象中。因此,派生类特有的成员变量和方法不会出现在基类对象中,就像它们被“切掉”了一样

在代码中:

class Student : public Person
{
public:int _No; // 学号
};
void Test()
{Student sobj;// 1.子类对象可以赋值给父类对象/指针/引用Person pobj = sobj;  // 切片发生在这里Person* pp = &sobj;  // 没有切片,因为 pp 指向的是一个 Student 对象Person& rp = sobj;   // 没有切片,因为 rp 引用的是一个 Student 对象
}
  • 在行 Person pobj = sobj; 中,由于 pobjPerson 类型的对象,sobj(一个 Student 对象)被赋值给 pobj 时,Student 类特有的 _No 成员被“切掉”,不会体现在 pobj 中。因此,pobj 中无法反映出 sobj 的完整状态和行为。

  • 在行 Person* pp = &sobj; 中,pp 是指向 Person 类型的指针,但它实际上指向了派生类 Student 的对象 sobj,没有发生切片,因为指针指向的是完整的 Student 对象。

  • 在行 Person& rp = sobj; 中,rp 是一个引用 Person 类型,它引用了 sobj,同样没有发生切片,因为引用关联的是 sobj 的完整实体。

实际上,在行 Person& rp = sobj; 中,引用 rp 的确是 Person 类型,但它并不导致对象切片。引用实际上并不拥有它所引用的对象,而只是提供另一个名称来访问现有对象。因此,当我们通过基类引用访问派生类对象时,并没有创建新的对象,也没有丢失派生类的任何部分。

在这行代码中:

Person& rp = sobj;

rp 实际上是对 sobj (它是一个 Student 类型的对象)的另一个访问方式。即使 rp 被声明为 Person 类型的引用,它实际引用的还是 sobj 的完整实体(包含 Person 部分和 Student 特有的部分)。但是,通过 rp 只能直接访问 sobj 中由 Person 定义的成员,Student 特有的成员(如 _No)不可以通过 rp 直接访问,除非进行了适当的强制转换

例子:

Person& rp = sobj;
rp._name = "Name";    // 可以访问,因为_name是Person的成员
// rp._No = 123;      // 错误!无法访问,因为_No是Student特有的成员,即使它实际上存在于sobj中

即使我们通过基类引用或指针操作对象,派生类对象的完整信息(所有成员变量和函数)仍然都在内存中,没有丢失。使用引用和指针时不会发生切片

对象切片的问题仅在派生类对象被赋值给另一个基类类型的对象时才会发生,比如当派生类对象被传值给一个基类对象的函数参数,或者通过赋值构造一个新的基类对象。这时候派生类特有的信息实际上会被切割掉并不会出现在新的基类对象中。在使用引用或指针时,这种情况并不会发生

  1. 基类对象不能赋值给派生类对象
  2. 基类的指针或者引用可以通过强制类型转换赋值给派生类的指针或者引用。但是必须是基类的指针是指向派生类对象时才是安全的。这里基类如果是多态类型,可以使用RTTI(Run-Time Type Information)的dynamic_cast 来进行识别后进行安全转换

3.继承中的作用域

  1. 在继承体系中基类和派生类都有独立的作用域
  2. 子类和父类中有同名成员,子类成员将屏蔽父类对同名成员的直接访问,这种情况叫隐藏,也叫重定义。(在子类成员函数中,可以使用 基类::基类成员 显示访问)
class Person
{
protected:string _name = "a"; // 姓名int _num = 111; // 身份证号
};
class Student : public Person
{
public:void Print(){cout << " 姓名:" << _name << endl;cout << " 身份证号:" << Person::_num << endl;cout << " 学号:" << _num << endl;}
protected:int _num = 999; // 学号
};
void Test()
{Student s1;s1.Print();
};

这段代码展示了成员隐藏,以及如何在派生类中访问基类的被隐藏成员的概念。

  • Student 类中,成员函数 Print 试图访问名称为 _num 的成员变量。由于派生类中存在同名成员,派生类的 _num 会隐藏基类的同名成员。

  • 如果在派生类中尝试访问一个被隐藏的基类成员,需要显式地使用类名限定符来指定基类的成员。在 Print 方法中使用 Person::_num 来访问基类 Person 中的 _num 成员。

输出结果将是:

姓名: a
身份证号: 111
学号: 999
  1. 需要注意的是如果是成员函数的隐藏,只需要函数名相同就构成隐藏
class A
{
public:void fun(){cout << "func()" << endl;}
};
class B : public A
{
public:void fun(int i){fun();cout << "func(int i)->" << i << endl;}
};

B中的fun和A中的fun 不是构成重载,因为不是在同一作用域
B中的fun和A中的fun 构成隐藏,成员函数满足函数名相同就构成隐藏

class B : public A
{
public:void fun(int i)  // 接受一个整型参数{fun();  // 编译器将会提示错误:找不到不带参数的 "fun" 函数。cout << "func(int i)->" << i << endl;}
};

在这个代码中,试图调用基类 Afun 函数。然而,由于派生类 B 提供了一个参数不同的版本 fun(int),所以基类 A 中的 fun 函数在派生类 B 的作用域中被隐藏了。C++ 规则规定,如果派生类提供了和基类同名的函数,基类中同名的函数在派生类的作用域就不再可见了

因此,在 B 类的成员函数 fun(int) 中,调用 fun() 试图无参数调用被隐藏的同名函数会无法编译,因为编译器认为我们试图调用 fun(int) 这个版本,但没有提供参数,导致参数不匹配

修复

为了调用基类 Afun 函数,我们必须显式地使用作用域解析运算符 :: 来指明我们想要调用的函数属于基类作用域:

class B : public A
{
public:void fun(int i){A::fun();  // 正确:调用基类 `A` 中的 `fun`cout << "func(int i)->" << i << endl;}
};

这样,当我们在类 Bfun(int i) 函数中调用 A::fun() 时,它将成功地调用基类 A 无参数的 fun 函数,然后输出整型参数 i 的值。

如果你希望在派生类中保留对基类中同名函数的访问能力(不希望隐藏),可以使用 using 声明在派生类中导入基类中的函数:

class B : public A
{
public:using A::fun;void fun(int i){fun();  // 正确:由于 "using A::fun;",此处调用的是基类 `A` 中的 `fun`cout << "func(int i)->" << i << endl;}
};

在实际编程中,为了避免混淆,通常不建议在派生类中使用与基类成员同名的变量。

本节内容到此结束!感谢大家阅读!

这篇关于【c++】继承学习(一):继承机制与基类派生类转换的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/957709

相关文章

关于C++中的虚拟继承的一些总结(虚拟继承,覆盖,派生,隐藏)

1.为什么要引入虚拟继承 虚拟继承是多重继承中特有的概念。虚拟基类是为解决多重继承而出现的。如:类D继承自类B1、B2,而类B1、B2都继承自类A,因此在类D中两次出现类A中的变量和函数。为了节省内存空间,可以将B1、B2对A的继承定义为虚拟继承,而A就成了虚拟基类。实现的代码如下: class A class B1:public virtual A; class B2:pu

C++对象布局及多态实现探索之内存布局(整理的很多链接)

本文通过观察对象的内存布局,跟踪函数调用的汇编代码。分析了C++对象内存的布局情况,虚函数的执行方式,以及虚继承,等等 文章链接:http://dev.yesky.com/254/2191254.shtml      论C/C++函数间动态内存的传递 (2005-07-30)   当你涉及到C/C++的核心编程的时候,你会无止境地与内存管理打交道。 文章链接:http://dev.yesky

51单片机学习记录———定时器

文章目录 前言一、定时器介绍二、STC89C52定时器资源三、定时器框图四、定时器模式五、定时器相关寄存器六、定时器练习 前言 一个学习嵌入式的小白~ 有问题评论区或私信指出~ 提示:以下是本篇文章正文内容,下面案例可供参考 一、定时器介绍 定时器介绍:51单片机的定时器属于单片机的内部资源,其电路的连接和运转均在单片机内部完成。 定时器作用: 1.用于计数系统,可

C++的模板(八):子系统

平常所见的大部分模板代码,模板所传的参数类型,到了模板里面,或实例化为对象,或嵌入模板内部结构中,或在模板内又派生了子类。不管怎样,最终他们在模板内,直接或间接,都实例化成对象了。 但这不是唯一的用法。试想一下。如果在模板内限制调用参数类型的构造函数会发生什么?参数类的对象在模板内无法构造。他们只能从模板的成员函数传入。模板不保存这些对象或者只保存他们的指针。因为构造函数被分离,这些指针在模板外

问题:第一次世界大战的起止时间是 #其他#学习方法#微信

问题:第一次世界大战的起止时间是 A.1913 ~1918 年 B.1913 ~1918 年 C.1914 ~1918 年 D.1914 ~1919 年 参考答案如图所示

[word] word设置上标快捷键 #学习方法#其他#媒体

word设置上标快捷键 办公中,少不了使用word,这个是大家必备的软件,今天给大家分享word设置上标快捷键,希望在办公中能帮到您! 1、添加上标 在录入一些公式,或者是化学产品时,需要添加上标内容,按下快捷键Ctrl+shift++就能将需要的内容设置为上标符号。 word设置上标快捷键的方法就是以上内容了,需要的小伙伴都可以试一试呢!

LangChain转换链:让数据处理更精准

1. 转换链的概念 在开发AI Agent(智能体)时,我们经常需要对输入数据进行预处理,这样可以更好地利用LLM。LangChain提供了一个强大的工具——转换链(TransformChain),它可以帮我们轻松实现这一任务。 转换链(TransformChain)主要是将 给定的数据 按照某个函数进行转换,再将 转换后的结果 输出给LLM。 所以转换链的核心是:根据业务逻辑编写合适的转换函

AssetBundle学习笔记

AssetBundle是unity自定义的资源格式,通过调用引擎的资源打包接口对资源进行打包成.assetbundle格式的资源包。本文介绍了AssetBundle的生成,使用,加载,卸载以及Unity资源更新的一个基本步骤。 目录 1.定义: 2.AssetBundle的生成: 1)设置AssetBundle包的属性——通过编辑器界面 补充:分组策略 2)调用引擎接口API

C++工程编译链接错误汇总VisualStudio

目录 一些小的知识点 make工具 可以使用windows下的事件查看器崩溃的地方 dumpbin工具查看dll是32位还是64位的 _MSC_VER .cc 和.cpp 【VC++目录中的包含目录】 vs 【C/C++常规中的附加包含目录】——头文件所在目录如何怎么添加,添加了以后搜索头文件就会到这些个路径下搜索了 include<> 和 include"" WinMain 和

Javascript高级程序设计(第四版)--学习记录之变量、内存

原始值与引用值 原始值:简单的数据即基础数据类型,按值访问。 引用值:由多个值构成的对象即复杂数据类型,按引用访问。 动态属性 对于引用值而言,可以随时添加、修改和删除其属性和方法。 let person = new Object();person.name = 'Jason';person.age = 42;console.log(person.name,person.age);//'J