大话C++语言:类继承与派生 c++类与类之间的继承关系
haoteby 2024-12-22 18:20 3 浏览
1 类继承的概述
类继承(Class Inheritance)是面向对象编程(Object-Oriented Programming,OOP)中的一个核心概念。它允许我们创建一个新的类(子类或派生类),该类继承了另一个已存在的类(父类或基类)的属性和方法。这样,子类可以重用父类的代码,而无需重新编写。
为什么会有类的继承呢?包括以下原因:
- 代码重用: 继承允许我们创建一个新的类,这个新类继承自一个已存在的类,从而重用那个类的代码。这意味着我们不需要从头开始编写所有代码,而是可以继承已有的代码,并在此基础上进行扩展或修改。这大大提高了代码的可重用性和开发效率。
- 扩展性: 通过继承,我们可以创建更具体的子类,这些子类不仅包含父类的功能,还可以添加新的功能或覆盖父类的功能。这使得我们能够创建更灵活、更适应特定需求的类。例如,我们可以有一个表示一般形状的基类(如Shape),然后创建继承自Shape的子类来表示特定的形状(如Circle、Rectangle等)。
- 多态性: 继承是多态性的一个关键组件。多态性意味着允许不同类型的对象对同一消息做出不同的响应。通过子类覆盖父类的方法,我们可以实现多态,这样当使用父类引用指向子类对象时,将调用子类的方法而不是父类的方法。这提供了更大的灵活性和可扩展性。
- 组织代码: 继承有助于组织代码,使其更加清晰和易于理解。通过将相似的功能或行为封装在基类中,并在需要时进行扩展或特化,我们可以创建一个结构良好、易于维护的代码库。
- 构建复杂的层次结构: 在现实世界中,很多事物都有层次结构关系。例如,动物是一个基类,而狗、猫等是动物类的子类。通过使用继承,我们可以模拟这种层次结构,并在代码中反映出来。这有助于创建更复杂、更逼真的模型。
- 促进软件设计原则: 继承是面向对象设计原则(如单一职责原则、开放封闭原则、里氏替换原则等)的重要实现手段。通过合理地使用继承,我们可以遵循这些原则,从而创建出更加健壮、可扩展和可维护的软件系统。
2 基类和派生类定义
2.1 基类定义
基类(Base Class)是一个被其他类继承的类。在C++中,基类的定义与其他类的定义非常相似,但它被设计为被其他类继承。基类通常包含一些公共的(public)和保护的(protected)成员变量和成员函数,这些成员可以被派生类访问和使用。基类提供了一种代码重用和扩展的机制,允许派生类继承基类的属性和行为,并添加或覆盖自己的特定实现。
基类的定义语法格式如下:
class BaseClass
{
public:
// 公共构造函数
BaseClass()
{
// 初始化代码
}
// 公共析构函数(通常声明为虚函数)
virtual ~BaseClass()
{
// 清理代码
}
// 公共成员函数
void publicFunction()
{
// 公共成员函数的实现
}
protected:
// 受保护的成员变量
int protectedVariable;
// 受保护的成员函数
void protectedFunction()
{
// 受保护成员函数的实现
}
private:
// 私有成员变量
int privateVariable;
// 私有成员函数
void privateFunction()
{
// 私有成员函数的实现
}
};
其中,
- virtual 关键字用于声明析构函数为虚函数。这允许在删除指向派生类对象的基类指针时,调用正确的析构函数(派生类的析构函数),从而正确释放资源。
- public 部分声明了公共构造函数 BaseClass() 和公共成员函数 publicFunction()。这些成员可以被任何对象(包括派生类对象)访问。
- protected 部分声明了一个受保护的成员变量 protectedVariable 和一个受保护的成员函数 protectedFunction()。这些成员可以被派生类访问,但不能被派生类对象或基类对象之外的其他代码访问。
- private 部分声明了一个私有成员变量 privateVariable 和一个私有成员函数 privateFunction()。这些成员只能被基类自身访问,不能被派生类或其他代码访问。
2.2 派生类定义
派生类(Derived Class)继承自一个或多个已有的类(称为基类或父类)。派生类继承了基类的所有公有(public)和保护(protected)成员(包括属性和方法),并可以添加新的成员或重写(override)基类的成员。
派生类的主要目的是实现代码的重用和扩展性。通过继承基类,派生类可以自动拥有基类中的所有非私有成员,这意味着派生类不需要重新实现这些成员,从而减少了代码量。同时,派生类还可以添加自己的新成员,以满足特定的需求。
派生类定义的语法格式如下:
// 基类定义
class BaseClass
{
public:
// 基类的构造函数
BaseClass();
// 基类的析构函数
virtual ~BaseClass();
// 基类的公有成员
void publicFunction();
protected:
// 基类的受保护成员
int protectedVariable;
private:
// 基类的私有成员
int privateVariable;
};
// 派生类定义
class DerivedClass : access_modifier BaseClass
{
public:
// 派生类的构造函数
DerivedClass();
// 派生类的析构函数
virtual ~DerivedClass();
// 派生类的公有成员
void anotherPublicFunction();
protected:
// 派生类的受保护成员
int anotherProtectedVariable;
private:
// 派生类的私有成员
int anotherPrivateVariable;
// 如果需要,可以在这里初始化基类成员
using BaseClass::baseMemberFunction; // 访问基类中的成员
// 重写基类中的虚函数
void virtualFunction() override;
};
// 派生类的构造函数实现
DerivedClass::DerivedClass() : BaseClass()
{
// 派生类构造函数的初始化列表,可以调用基类的构造函数
// 初始化派生类特有的成员
}
// 派生类的析构函数实现
DerivedClass::~DerivedClass()
{
// 清理派生类特有的资源
}
// 派生类成员函数的实现
void DerivedClass::anotherPublicFunction()
{
// 实现派生类的另一个公有成员函数
}
// 重写基类虚函数的实现
void DerivedClass::virtualFunction()
{
// 实现重写的虚函数
}
其中, access_modifier指定继承方式,包括public、protected 或 private三种方式。
2.3 综合案例
我们先定义一个形状的Shape基类,它包含一些通用的属性和方法。然后,将从这个基类派生出Circle和Rectangle类,它们分别表示圆形和矩形,并添加各自特有的属性和方法。
#include <iostream>
#include <string>
// 基类Shape定义
class Shape
{
public:
// 构造函数
Shape(const std::string& color) : color(color)
{
}
// 析构函数
virtual ~Shape()
{
}
// 公有成员函数
virtual void Draw() const
{
std::cout << "Drawing a shape of color " << color << std::endl;
}
virtual double GetArea() const
{
return 0.0; // 基类中的默认实现,可能会被派生类重写
}
protected:
// 受保护成员变量
std::string color;
};
// 派生类圆形类Circle定义
class Circle : public Shape
{
public:
// 构造函数
Circle(double r, const std::string& color) :Shape(color), radius(r)
{
}
// 析构函数
virtual ~Circle()
{
}
// 重写基类的draw函数
void Draw() const override
{
std::cout << "Drawing a circle with radius " << radius
<< " and color " << color << std::endl;
}
// 重写基类的getArea函数
double GetArea() const override
{
return 3.14 * radius * radius; // 近似计算圆的面积
}
private:
// 私有成员变量
double radius;
};
// 派生类矩形类Rectangle定义
class Rectangle : public Shape
{
public:
// 构造函数
Rectangle(double w, double h, const std::string& color)
: Shape(color), width(w), height(h)
{
}
// 重写基类的draw函数
void Draw() const override
{
std::cout << "Drawing a rectangle with width " << width
<< " and height " << height
<< " and color " << color << std::endl;
}
// 重写基类的getArea函数
double GetArea() const override
{
return width * height;
}
private:
// 私有成员变量
double width;
double height;
};
int main()
{
// 创建圆形和矩形的对象
Circle circle(5, "red");
Rectangle rectangle(4, 6, "blue");
// 调用对象的成员函数
circle.Draw();
std::cout << "Circle area: " << circle.GetArea() << std::endl;
rectangle.Draw();
std::cout << "Rectangle area: " << rectangle.GetArea() << std::endl;
return 0;
}
其中,
- Shape类提供了Draw和GetArea方法的默认实现。
- 派生类Circle和Rectangle通过重写(override关键字)这些方法提供特定的实现。
- 如果基类Shape中的Draw和GetArea不是虚函数(virtual关键字定义),那么,派生类Circle和Rectangle中使用了override关键字重写这些函数时,编译器会报错。
扩展知识:
在C++11及后续版本中,override关键字用于显式指示一个成员函数打算重写(override)基类中的虚函数。使用override关键字可以帮助编译器检查你的意图是否正确:如果基类中没有你要重写的虚函数,编译器将产生一个错误。
override关键字增加了代码的可读性和安全性,因为它明确表明了你正在重写基类中的某个函数,而不是创建一个新的、可能名称相似的函数。如果基类中的虚函数在未来的某个版本中被删除或更改了签名,使用override关键字的代码将不会编译通过,从而提醒开发者需要更新代码。
---E N D---
喜欢的记得关注哦!
您的支持是我们前进的动力!
相关推荐
- 单点登录(SSO)解决方案介绍(单点登录概念)
-
一、单点登录的介绍单点登录(SingleSignOn),简称为SSO,是目前比较流行的企业业务整合的解决方案之一。SSO的定义是在多个应用系统中,用户只需要登录一次就可以访问所有相互信任的应用系...
- 系统登录的三种方式,哪一种更安全?
-
登录是一个高频的动作,笔者抓住这一个小点,分析了系统登录的几种方式和对应的场景。今天谈谈登录。登录即用户输入用户名和密码登录进系统中。B端系统,对于登录的业务场景有两种(可能不止,目前遇到过这两种):...
- 到底什么是单点登录(SSO)?(什么叫做单点登录)
-
什么是单点登录?单点登录(SingleSign-On,简称SSO)是一种集中式的身份验证和授权机制,用户只需在一处输入一次凭证(例如用户名和密码)就可以访问多个相关但独立的软件系统。在数字化时代,...
- 5年稳如老狗的单点登录系统,到底是怎么搞出来的?
-
说到单点登录(SingleSign-On,简称SSO),大家的第一反应可能是——啊不就是登录一次,能到处串门儿嘛?别说,还真差不多,就是这么个意思。但真要搭一套好用、耐造、还能扛住公司里各种奇奇怪...
- 这些负载均衡都解决哪些问题?服务、网关、NGINX?
-
在微服务项目中,有服务的负载均衡、网关的负载均衡、Nginx的负载均衡,这几个负载均衡分别用来解决什么问题呢?一、服务的负载均衡先抛出一个问题:...
- Nginx负载均衡最全详解(4大算法原理机制)
-
Nginx在大型网站架构很重要,也是大厂重点考察方向,今天我就重点来详解Nginx负载均衡@mikechen本篇已收于mikechen原创超30万字《阿里架构师进阶专题合集》里面。Nginx负载均衡N...
- 负载均衡 Nginx Session 一致性(nginx 负载均衡 会话保持)
-
HTTPS请求跳转...
- 监控Oracle Cloud负载均衡器:Applications Manager释放最佳性能
-
设想你正在运营一个受欢迎的在线学习平台,在考试前的高峰期,平台流量激增。全球的学生同时登录,观看视频、提交作业和参加测试。如果OracleCloud负载均衡器不能高效地分配流量,或者后端服务器难...
- Nginx负载均衡:nginx.conf配置文件说明!
-
大家好,欢迎来到程序视点!我是你们的老朋友.小二!在此记录下Nginx服务器nginx.conf负载均衡的配置文件说明,部分注释收集与网络.关于nginx.conf基本的配置,请查看上一篇文章!Ng...
- Java高可用系统架构中的负载均衡策略
-
Java高可用系统架构中的负载均衡策略在现代的分布式系统中,负载均衡策略是构建高可用系统的基石。Java开发者需要深刻理解这些策略,以便打造稳定且高效的系统。接下来,让我们一起揭开负载均衡的神秘面纱。...
- Spring Boot3 客户端负载均衡全解析:从原理到实战
-
在当今互联网大厂后端技术开发的激烈竞争环境中,构建高效、稳定的微服务架构是核心诉求。其中,SpringBoot3作为热门开发框架,其客户端负载均衡功能对于提升系统性能、保障服务稳定性起着关键作用。...
- MySql高可用集群MySQL Router负载均衡读写分离
-
名词解释MGR:MysqlGroupReplication组复制,多台MySQL服务器在同一组中会自动保持同步状态,当某台服务器故障时,整个复制组依然可以保持正常并对外提供服务。...
- 性能测试之tomcat+nginx负载均衡(nginx tomcat)
-
nginxtomcat配置准备工作:两个tomcat执行命令cp-rapache-tomcat-8.5.56apache-tomcat-8.5.56_2修改被复制的tomcat2下con...
- win10/11双网卡链路聚合叠加负载均衡提升网速解决网卡网速瓶颈!
-
双网卡链路聚合一种网络配置技术,通过将多个物理网卡绑定在一起,形成一个逻辑上的网络接口,以提高网络的可靠性、可用性和性能。这种技术通常用于服务器和网络设备中,以实现负载均衡、冗余和高可用性。本机环境:...