【C++面向对象——类的多态性与虚函数】计算图像面积(头歌实践教学平台习题)【合集】

简介: 本任务要求设计一个矩形类、圆形类和图形基类,计算并输出相应图形面积。相关知识点包括纯虚函数和抽象类的使用。**目录:**- 任务描述- 相关知识 - 纯虚函数 - 特点 - 使用场景 - 作用 - 注意事项 - 相关概念对比 - 抽象类的使用 - 定义与概念 - 使用场景- 编程要求- 测试说明- 通关代码- 测试结果**任务概述:**1. **图形基类(Shape)**:包含纯虚函数 `void PrintArea()`。2. **矩形类(Rectangle)**:继承 Shape 类,重写 `Print

目录😋

任务描述

相关知识

1. 纯虚函数

一、特点

二、使用场景

三、作用

四、注意事项

五、相关概念对比

2. 抽象类的使用

一、定义与概念

二、使用场景

编程要求

测试说明

通关代码

测试结果


任务描述

本关任务:设计一个矩形类、一个圆形类和一个图形基类,计算并输出相应图形面积。

相关知识

为了完成本关任务,你需要掌握:

  1. 纯虚函数
  2. 抽象类的使用

1. 纯虚函数

一、特点

  1. 函数声明形式
    纯虚函数在声明时有其特定的语法形式,如 virtual 函数类型 函数名(参数列表) = 0;。以之前提到的 Base 类中的 virtual void Func() = 0; 为例,virtual 关键字表明这是一个虚函数,而最后的 = 0 则明确指出它是纯虚函数,意味着该函数在当前类(这里是 Base 类)中不提供具体的函数实现(也就是没有函数体),仅预留函数名和参数列表等信息,等待派生类去完善其具体功能。
  2. 不可直接调用
    由于纯虚函数本身没有函数体,在基类层面它是不能被直接调用的。如果尝试在基类对象上调用纯虚函数,编译器会报错,因为它没有实际可执行的代码逻辑与之对应。例如:
Base baseObj;
baseObj.Func();  // 这样的调用会导致编译错误,因为Base类中Func是纯虚函数,无函数体
  1. image.gif
  2. 派生类要求
    纯虚函数必须在派生类中进行定义,否则该虚函数在派生类中依然保持为纯虚函数状态。当派生类实现了这个纯虚函数后,才能通过派生类的对象调用这个函数,且调用时执行的是派生类中定义的函数逻辑。例如:
class Derived : public Base {
public:
    void Func() override {
        // 在这里定义Func函数在Derived类中的具体实现逻辑
        std::cout << "This is the implementation of Func in Derived class." << std::endl;
    }
};
Derived derivedObj;
derivedObj.Func();  // 调用Derived类中实现的Func函数,输出相应内容,编译通过
  1. image.gif

二、使用场景

  1. 实现多态性
    纯虚函数是实现面向对象编程中多态性的重要手段之一。基类定义了一系列纯虚函数作为接口,不同的派生类根据自身的特点和需求去具体实现这些函数,这样就可以通过基类指针或引用指向不同的派生类对象,并调用这些虚函数时,执行不同派生类中对应的函数实现,呈现出多态的行为。例如,设计一个图形类作为基类,有 draw() 这样的纯虚函数,然后派生出 Circle(圆形)、Rectangle(矩形)等具体图形类,每个派生类各自实现 draw() 函数来绘制对应的图形,通过基类指针可以统一操作不同图形对象的绘制操作,代码可能如下:
class Shape {
public:
    virtual void draw() = 0;
};
class Circle : public Shape {
public:
    void draw() override {
        std::cout << "Drawing a circle." << std::endl;
    }
};
class Rectangle : public Shape {
public:
    void draw() override {
        std::cout << "Drawing a rectangle." << std::endl;
    }
};
int main() {
    Shape* shapePtr1 = new Circle();
    Shape* shapePtr2 = new Rectangle();
    shapePtr1->draw();
    shapePtr2->draw();
    delete shapePtr1;
    delete shapePtr2;
    return 0;
}
  1. image.gif 在上述代码中,通过基类 Shape 的指针指向不同派生类对象,调用 draw 函数时,根据对象实际类型执行相应派生类中定义的绘制逻辑,体现了多态性。
  2. 定义抽象类
    含有纯虚函数的类被称为抽象类,抽象类不能实例化对象(像前面例子中直接创建 Base 类对象就会报错),它更多的是作为一种抽象的概念和接口规范存在,用于为派生类提供统一的函数接口框架,引导派生类去实现特定的功能,使得整个类层次结构在设计上更加清晰、规范,便于代码的扩展和维护。例如在设计一个动物类层次结构时,动物都有 makeSound() 这个行为,但不同动物发声方式不同,所以可以在基类 Animal 中定义 virtual void makeSound() = 0; 纯虚函数,然后各种具体动物类(如 DogCat 等)去实现这个函数来体现各自独特的叫声。

三、作用

  1. 接口规范作用
    纯虚函数在基类中定义了一个统一的函数接口,明确告知派生类需要实现哪些功能,保证了派生类在实现相关功能时有一致的函数签名(函数名、参数列表、返回类型等方面符合基类定义),有助于提高代码的可读性和可维护性。不同的开发人员在编写派生类时,能清楚知道需要遵循的接口规范,避免随意编写函数而导致代码逻辑混乱。
  2. 代码扩展性提升
    当需要在已有类层次结构基础上添加新的派生类时,只要按照基类中纯虚函数定义的接口去实现相应功能即可,不需要对基类或者其他已有派生类做大量修改。例如,在前面图形类的例子中,如果后续要添加一个 Triangle(三角形)类,只需从 Shape 类派生,然后实现 draw 函数来绘制三角形就行,原有操作图形绘制的代码(通过基类指针调用 draw 函数的部分)无需改动就能适用于新的三角形图形对象,方便了代码的扩展和功能的丰富。

四、注意事项

  1. 继承关系中的纯虚函数处理
    在多层继承结构中,如果中间层派生类没有对基类的纯虚函数进行定义,那么这个纯虚函数依然会传递下去,在该中间层派生类的派生类中依然需要进行定义才能实例化对象。例如:
class Base {
public:
    virtual void func() = 0;
};
class Intermediate : public Base {
// 这里没有对func函数进行定义,func在Intermediate类中依然是纯虚函数
};
class Derived : public Intermediate {
public:
    void func() override {
        // 在这里定义func函数实现,使得Derived类可以实例化对象
    }
};
  1. image.gif
  2. 函数签名一致性
    派生类在定义纯虚函数时,必须保证函数签名(包括函数名、参数列表、返回类型,返回类型协变情况除外)与基类中纯虚函数的定义严格一致,否则编译器会认为是在重新定义一个新的函数,而不是实现基类中的纯虚函数,导致编译错误。例如:
class Base {
public:
    virtual int calculate(int num) = 0;
};
class Derived : public Base {
public:
    double calculate(int num) override {  // 返回类型不一致,会导致编译错误
        return 0.0;
    }
};
  1. image.gif

五、相关概念对比

  1. 虚函数与纯虚函数
    虚函数可以在基类中有具体的函数体实现,派生类可以选择重写(override)它来实现多态性,也可以不重写而直接继承基类的函数实现。而纯虚函数在基类中没有函数体,必须由派生类去定义实现,主要用于定义抽象类和接口规范,引导派生类进行特定功能的实现,以此来实现多态等面向对象编程特性。
  2. 抽象类与具体类
    含有纯虚函数的类就是抽象类,它不能被实例化,侧重于提供抽象的接口和概念框架。而具体类是可以实例化对象的类,通常是在抽象类基础上,通过实现其纯虚函数等方式,完善了具体功能,从而成为能够创建实际对象并使用的类,比如前面例子中的 CircleRectangle 等就是具体类,它们基于抽象的 Shape 类实现了具体绘制图形的功能,进而可以创建相应图形对象进行操作。

2. 抽象类的使用

一、定义与概念

   抽象类是一种不能被实例化的类,它通常包含一个或多个纯虚函数。纯虚函数是在声明时被初始化为 0 的虚函数,例如:virtual void func() = 0;。抽象类主要用于为派生类提供一个通用的接口规范,定义一系列的行为和属性,但把具体的实现细节留给派生类。

 

二、使用场景

1、多态性实现

       假设要开发一个图形绘制程序,有多种图形如圆形、矩形等。可以先定义一个抽象类Shape作为基类,其中包含一个纯虚函数draw():

class Shape {
public:
    virtual void draw() = 0;
};
image.gif

然后派生出具体的图形类,如CircleRectangle,它们分别实现draw()函数来绘制自己的形状:

class Circle : public Shape {
public:
    void draw() override {
        // 绘制圆形的具体代码,比如使用绘图库来绘制
        cout << "Drawing a circle." << endl;
    }
};
class Rectangle : public Shape {
public:
    void draw() override {
        // 绘制矩形的具体代码
        cout << "Drawing a rectangle." << endl;
    }
};
image.gif

这样,通过基类指针或引用,可以方便地调用不同派生类的draw()函数,实现多态性。例如:

Shape* shapePtr;
shapePtr = new Circle();
shapePtr->draw();
delete shapePtr;
shapePtr = new Rectangle();
shapePtr->draw();
delete shapePtr;
image.gif

2、代码结构规范:

       在大型项目中,抽象类可以用于规范代码结构。比如在一个游戏开发项目中,有不同类型的角色,如战士、法师等。可以定义一个抽象类Character,其中包含纯虚函数attack()move()等:

class Character {
public:
    virtual void attack() = 0;
    virtual void move() = 0;
};
image.gif

战士类Warrior和法师类Mage等派生类实现这些纯虚函数来定义自己的攻击和移动方式。这种方式使得不同角色的行为定义有了统一的规范,便于代码的维护和扩展。例如:

class Warrior : public Character {
public:
    void attack() override {
        cout << "Warrior attacks with sword." << endl;
    }
    void move() override {
        cout << "Warrior runs quickly." << endl;
    }
};
class Mage : public Character {
public:
    void attack() override {
        cout << "Mage casts a spell." << endl;
    }
    void move() override {
        cout << "Mage teleports." << endl;
    }
};
image.gif

3、继承关系中的注意事项

  • 纯虚函数的继承:如果基类是抽象类,派生类没有实现基类中的所有纯虚函数,那么派生类仍然是抽象类。例如:
class Base {
public:
    virtual void func1() = 0;
    virtual void func2() = 0;
};
class Derived : public Base {
public:
    void func1() override {
        // 实现func1
    }
};
  • image.gif 在这个例子中,Derived类仍然是抽象类,因为它没有实现func2,所以不能实例化Derived类的对象。
  • 函数签名一致性:派生类在实现抽象类中的纯虚函数时,要保证函数签名(包括函数名、参数类型、返回类型等)与抽象类中的定义一致。只有满足这个条件,才能正确地实现多态性。例如,如果抽象类中有一个纯虚函数virtual int calculate(double num) = 0;,派生类中实现的函数应该具有相同的函数名、参数类型和返回类型,如int calculate(double num) override {... }

4、与具体类的对比

       具体类是可以直接实例化对象的类,它实现了所有必要的功能。而抽象类侧重于定义接口和行为规范,不能直接创建对象。抽象类为具体类提供了一个模板,具体类通过继承抽象类并实现其纯虚函数来具体化抽象类所定义的概念。例如,在前面图形的例子中,CircleRectangle是具体类,可以创建它们的对象来表示具体的图形并进行绘制操作,而Shape是抽象类,用于规定所有图形类都应该有draw()这个行为,但本身不能用来创建一个没有具体形状的图形对象。


编程要求

在右侧编辑器中的Begin-End之间补充代码,设计图像基类、矩形类和圆形类三个类,函数成员变量据情况自己拟定,其他要求如下:

  • 图形类(Shape)
  • 纯虚函数:void PrintArea(),用于输出当前图形的面积。
  • 矩形类(Rectangle)
  • 继承关系:继承 Shape 类,并且重写 PrintArea 函数,输出矩形的面积,输出格式为:矩形面积 = width*height。
  • 带参构造函数:Rectangle(float w,float h),这两个参数分别赋值给成员变量的宽、高。
  • 圆形类(Circle)
  • 继承关系:继承 Shape 类,并且重写 PrintArea 函数,输出圆形的面积,输出格式为:圆形面积 = radio * radio * 3.14。
  • 带参构造函数:Circle(float r),参数 r 代表圆的半径。

测试说明

平台会对你编写的代码进行测试,比对你输出的数值与实际正确数值,只有所有数据全部计算正确才能通过测试:

测试输入:

10 2.5

预期输出:

矩形面积 = 20
圆形面积 = 314
image.gif

测试输入:

2 2.5

预期输出:

矩形面积 = 4
圆形面积 = 12.56
image.gif

开始你的任务吧,祝你成功!


通关代码

#include <iostream>
using namespace std;
/********* Begin *********/
class Shape
{
    //基类的声明
    
   public:
   virtual void PrintArea() = 0;
    
};
class Rectangle : public Shape
{
    //矩形类的声明
    public:
    void PrintArea()override;
    float width;
    float height;
    Rectangle(float w,float h);
};
//矩形类的定义
void Rectangle::PrintArea(){
    cout<<"矩形面积 = "<<width * height<<endl;
}
Rectangle::Rectangle(float w,float h){
    width = w;
    height = h;
}
class Circle : public Shape
{
    //圆形类的声明
    public:
    void PrintArea()override;
    float radio;
    Circle(float r);
};
//圆形类的定义
void Circle::PrintArea(){
    cout <<"圆形面积 = "<<radio * radio *3.14<<endl;
}
Circle::Circle(float r){
    radio = r;
}
/********* End *********/

image.gif


测试结果

image.gif

目录
相关文章
|
15天前
|
存储 C语言 C++
【C++数据结构——栈与队列】顺序栈的基本运算(头歌实践教学平台习题)【合集】
本关任务:编写一个程序实现顺序栈的基本运算。开始你的任务吧,祝你成功!​ 相关知识 初始化栈 销毁栈 判断栈是否为空 进栈 出栈 取栈顶元素 1.初始化栈 概念:初始化栈是为栈的使用做准备,包括分配内存空间(如果是动态分配)和设置栈的初始状态。栈有顺序栈和链式栈两种常见形式。对于顺序栈,通常需要定义一个数组来存储栈元素,并设置一个变量来记录栈顶位置;对于链式栈,需要定义节点结构,包含数据域和指针域,同时初始化栈顶指针。 示例(顺序栈): 以下是一个简单的顺序栈初始化示例,假设用C语言实现,栈中存储
132 77
|
15天前
|
存储 C++
【C++数据结构——树】哈夫曼树(头歌实践教学平台习题) 【合集】
【数据结构——树】哈夫曼树(头歌实践教学平台习题)【合集】目录 任务描述 相关知识 测试说明 我的通关代码: 测试结果:任务描述 本关任务:编写一个程序构建哈夫曼树和生成哈夫曼编码。 相关知识 为了完成本关任务,你需要掌握: 1.如何构建哈夫曼树, 2.如何生成哈夫曼编码。 测试说明 平台会对你编写的代码进行测试: 测试输入: 1192677541518462450242195190181174157138124123 (用户分别输入所列单词的频度) 预
52 14
【C++数据结构——树】哈夫曼树(头歌实践教学平台习题) 【合集】
|
15天前
|
C++ 芯片
【C++面向对象——类与对象】Computer类(头歌实践教学平台习题)【合集】
声明一个简单的Computer类,含有数据成员芯片(cpu)、内存(ram)、光驱(cdrom)等等,以及两个公有成员函数run、stop。只能在类的内部访问。这是一种数据隐藏的机制,用于保护类的数据不被外部随意修改。根据提示,在右侧编辑器补充代码,平台会对你编写的代码进行测试。成员可以在派生类(继承该类的子类)中访问。成员,在类的外部不能直接访问。可以在类的外部直接访问。为了完成本关任务,你需要掌握。
55 19
|
15天前
|
C++
【C++数据结构——树】二叉树的性质(头歌实践教学平台习题)【合集】
本文档介绍了如何根据二叉树的括号表示串创建二叉树,并计算其结点个数、叶子结点个数、某结点的层次和二叉树的宽度。主要内容包括: 1. **定义二叉树节点结构体**:定义了包含节点值、左子节点指针和右子节点指针的结构体。 2. **实现构建二叉树的函数**:通过解析括号表示串,递归地构建二叉树的各个节点及其子树。 3. **使用示例**:展示了如何调用 `buildTree` 函数构建二叉树并进行简单验证。 4. **计算二叉树属性**: - 计算二叉树节点个数。 - 计算二叉树叶子节点个数。 - 计算某节点的层次。 - 计算二叉树的宽度。 最后,提供了测试说明及通关代
38 10
|
15天前
|
存储 编译器 数据安全/隐私保护
【C++面向对象——类与对象】CPU类(头歌实践教学平台习题)【合集】
声明一个CPU类,包含等级(rank)、频率(frequency)、电压(voltage)等属性,以及两个公有成员函数run、stop。根据提示,在右侧编辑器补充代码,平台会对你编写的代码进行测试。​ 相关知识 类的声明和使用。 类的声明和对象的声明。 构造函数和析构函数的执行。 一、类的声明和使用 1.类的声明基础 在C++中,类是创建对象的蓝图。类的声明定义了类的成员,包括数据成员(变量)和成员函数(方法)。一个简单的类声明示例如下: classMyClass{ public: int
39 13
|
15天前
|
编译器 数据安全/隐私保护 C++
【C++面向对象——继承与派生】派生类的应用(头歌实践教学平台习题)【合集】
本实验旨在学习类的继承关系、不同继承方式下的访问控制及利用虚基类解决二义性问题。主要内容包括: 1. **类的继承关系基础概念**:介绍继承的定义及声明派生类的语法。 2. **不同继承方式下对基类成员的访问控制**:详细说明`public`、`private`和`protected`继承方式对基类成员的访问权限影响。 3. **利用虚基类解决二义性问题**:解释多继承中可能出现的二义性及其解决方案——虚基类。 实验任务要求从`people`类派生出`student`、`teacher`、`graduate`和`TA`类,添加特定属性并测试这些类的功能。最终通过创建教师和助教实例,验证代码
42 5
|
15天前
|
存储 算法 搜索推荐
【C++面向对象——群体类和群体数据的组织】实现含排序功能的数组类(头歌实践教学平台习题)【合集】
1. **相关排序和查找算法的原理**:介绍直接插入排序、直接选择排序、冒泡排序和顺序查找的基本原理及其实现代码。 2. **C++ 类与成员函数的定义**:讲解如何定义`Array`类,包括类的声明和实现,以及成员函数的定义与调用。 3. **数组作为类的成员变量的处理**:探讨内存管理和正确访问数组元素的方法,确保在类中正确使用动态分配的数组。 4. **函数参数传递与返回值处理**:解释排序和查找函数的参数传递方式及返回值处理,确保函数功能正确实现。 通过掌握这些知识,可以顺利地将排序和查找算法封装到`Array`类中,并进行测试验证。编程要求是在右侧编辑器补充代码以实现三种排序算法
33 5
|
15天前
|
设计模式 IDE 编译器
【C++面向对象——类的多态性与虚函数】编写教学游戏:认识动物(头歌实践教学平台习题)【合集】
本项目旨在通过C++编程实现一个教学游戏,帮助小朋友认识动物。程序设计了一个动物园场景,包含Dog、Bird和Frog三种动物。每个动物都有move和shout行为,用于展示其特征。游戏随机挑选10个动物,前5个供学习,后5个用于测试。使用虚函数和多态实现不同动物的行为,确保代码灵活扩展。此外,通过typeid获取对象类型,并利用strstr辅助判断类型。相关头文件如&lt;string&gt;、&lt;cstdlib&gt;等确保程序正常运行。最终,根据小朋友的回答计算得分,提供互动学习体验。 - **任务描述**:编写教学游戏,随机挑选10个动物进行展示与测试。 - **类设计**:基类
27 3
|
2月前
|
存储 编译器 C语言
【c++丨STL】string类的使用
本文介绍了C++中`string`类的基本概念及其主要接口。`string`类在C++标准库中扮演着重要角色,它提供了比C语言中字符串处理函数更丰富、安全和便捷的功能。文章详细讲解了`string`类的构造函数、赋值运算符、容量管理接口、元素访问及遍历方法、字符串修改操作、字符串运算接口、常量成员和非成员函数等内容。通过实例演示了如何使用这些接口进行字符串的创建、修改、查找和比较等操作,帮助读者更好地理解和掌握`string`类的应用。
81 2
|
2月前
|
存储 编译器 C++
【c++】类和对象(下)(取地址运算符重载、深究构造函数、类型转换、static修饰成员、友元、内部类、匿名对象)
本文介绍了C++中类和对象的高级特性,包括取地址运算符重载、构造函数的初始化列表、类型转换、static修饰成员、友元、内部类及匿名对象等内容。文章详细解释了每个概念的使用方法和注意事项,帮助读者深入了解C++面向对象编程的核心机制。
141 5