【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

相关文章

JAVA中整型数组、字符串数组、整型数和字符串 的创建与转换的方法

《JAVA中整型数组、字符串数组、整型数和字符串的创建与转换的方法》本文介绍了Java中字符串、字符数组和整型数组的创建方法,以及它们之间的转换方法,还详细讲解了字符串中的一些常用方法,如index... 目录一、字符串、字符数组和整型数组的创建1、字符串的创建方法1.1 通过引用字符数组来创建字符串1.2

深入理解C++ 空类大小

《深入理解C++空类大小》本文主要介绍了C++空类大小,规定空类大小为1字节,主要是为了保证对象的唯一性和可区分性,满足数组元素地址连续的要求,下面就来了解一下... 目录1. 保证对象的唯一性和可区分性2. 满足数组元素地址连续的要求3. 与C++的对象模型和内存管理机制相适配查看类对象内存在C++中,规

Spring使用@Retryable实现自动重试机制

《Spring使用@Retryable实现自动重试机制》在微服务架构中,服务之间的调用可能会因为一些暂时性的错误而失败,例如网络波动、数据库连接超时或第三方服务不可用等,在本文中,我们将介绍如何在Sp... 目录引言1. 什么是 @Retryable?2. 如何在 Spring 中使用 @Retryable

在 VSCode 中配置 C++ 开发环境的详细教程

《在VSCode中配置C++开发环境的详细教程》本文详细介绍了如何在VisualStudioCode(VSCode)中配置C++开发环境,包括安装必要的工具、配置编译器、设置调试环境等步骤,通... 目录如何在 VSCode 中配置 C++ 开发环境:详细教程1. 什么是 VSCode?2. 安装 VSCo

Java将时间戳转换为Date对象的方法小结

《Java将时间戳转换为Date对象的方法小结》在Java编程中,处理日期和时间是一个常见需求,特别是在处理网络通信或者数据库操作时,本文主要为大家整理了Java中将时间戳转换为Date对象的方法... 目录1. 理解时间戳2. Date 类的构造函数3. 转换示例4. 处理可能的异常5. 考虑时区问题6.

C++11的函数包装器std::function使用示例

《C++11的函数包装器std::function使用示例》C++11引入的std::function是最常用的函数包装器,它可以存储任何可调用对象并提供统一的调用接口,以下是关于函数包装器的详细讲解... 目录一、std::function 的基本用法1. 基本语法二、如何使用 std::function

基于C#实现将图片转换为PDF文档

《基于C#实现将图片转换为PDF文档》将图片(JPG、PNG)转换为PDF文件可以帮助我们更好地保存和分享图片,所以本文将介绍如何使用C#将JPG/PNG图片转换为PDF文档,需要的可以参考下... 目录介绍C# 将单张图片转换为PDF文档C# 将多张图片转换到一个PDF文档介绍将图片(JPG、PNG)转

HarmonyOS学习(七)——UI(五)常用布局总结

自适应布局 1.1、线性布局(LinearLayout) 通过线性容器Row和Column实现线性布局。Column容器内的子组件按照垂直方向排列,Row组件中的子组件按照水平方向排列。 属性说明space通过space参数设置主轴上子组件的间距,达到各子组件在排列上的等间距效果alignItems设置子组件在交叉轴上的对齐方式,且在各类尺寸屏幕上表现一致,其中交叉轴为垂直时,取值为Vert

Ilya-AI分享的他在OpenAI学习到的15个提示工程技巧

Ilya(不是本人,claude AI)在社交媒体上分享了他在OpenAI学习到的15个Prompt撰写技巧。 以下是详细的内容: 提示精确化:在编写提示时,力求表达清晰准确。清楚地阐述任务需求和概念定义至关重要。例:不用"分析文本",而用"判断这段话的情感倾向:积极、消极还是中性"。 快速迭代:善于快速连续调整提示。熟练的提示工程师能够灵活地进行多轮优化。例:从"总结文章"到"用

JVM 的类初始化机制

前言 当你在 Java 程序中new对象时,有没有考虑过 JVM 是如何把静态的字节码(byte code)转化为运行时对象的呢,这个问题看似简单,但清楚的同学相信也不会太多,这篇文章首先介绍 JVM 类初始化的机制,然后给出几个易出错的实例来分析,帮助大家更好理解这个知识点。 JVM 将字节码转化为运行时对象分为三个阶段,分别是:loading 、Linking、initialization