【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

目录
相关文章
|
18小时前
|
设计模式 IDE 编译器
【C++面向对象——类的多态性与虚函数】编写教学游戏:认识动物(头歌实践教学平台习题)【合集】
本项目旨在通过C++编程实现一个教学游戏,帮助小朋友认识动物。程序设计了一个动物园场景,包含Dog、Bird和Frog三种动物。每个动物都有move和shout行为,用于展示其特征。游戏随机挑选10个动物,前5个供学习,后5个用于测试。使用虚函数和多态实现不同动物的行为,确保代码灵活扩展。此外,通过typeid获取对象类型,并利用strstr辅助判断类型。相关头文件如&lt;string&gt;、&lt;cstdlib&gt;等确保程序正常运行。最终,根据小朋友的回答计算得分,提供互动学习体验。 - **任务描述**:编写教学游戏,随机挑选10个动物进行展示与测试。 - **类设计**:基类
10 3
|
15小时前
|
编译器 数据安全/隐私保护 C++
【C++面向对象——继承与派生】派生类的应用(头歌实践教学平台习题)【合集】
本实验旨在学习类的继承关系、不同继承方式下的访问控制及利用虚基类解决二义性问题。主要内容包括: 1. **类的继承关系基础概念**:介绍继承的定义及声明派生类的语法。 2. **不同继承方式下对基类成员的访问控制**:详细说明`public`、`private`和`protected`继承方式对基类成员的访问权限影响。 3. **利用虚基类解决二义性问题**:解释多继承中可能出现的二义性及其解决方案——虚基类。 实验任务要求从`people`类派生出`student`、`teacher`、`graduate`和`TA`类,添加特定属性并测试这些类的功能。最终通过创建教师和助教实例,验证代码
17 5
|
15小时前
|
存储 编译器 数据安全/隐私保护
【C++面向对象——类与对象】CPU类(头歌实践教学平台习题)【合集】
声明一个CPU类,包含等级(rank)、频率(frequency)、电压(voltage)等属性,以及两个公有成员函数run、stop。根据提示,在右侧编辑器补充代码,平台会对你编写的代码进行测试。​ 相关知识 类的声明和使用。 类的声明和对象的声明。 构造函数和析构函数的执行。 一、类的声明和使用 1.类的声明基础 在C++中,类是创建对象的蓝图。类的声明定义了类的成员,包括数据成员(变量)和成员函数(方法)。一个简单的类声明示例如下: classMyClass{ public: int
27 13
|
14小时前
|
C++ 芯片
【C++面向对象——类与对象】Computer类(头歌实践教学平台习题)【合集】
声明一个简单的Computer类,含有数据成员芯片(cpu)、内存(ram)、光驱(cdrom)等等,以及两个公有成员函数run、stop。只能在类的内部访问。这是一种数据隐藏的机制,用于保护类的数据不被外部随意修改。根据提示,在右侧编辑器补充代码,平台会对你编写的代码进行测试。成员可以在派生类(继承该类的子类)中访问。成员,在类的外部不能直接访问。可以在类的外部直接访问。为了完成本关任务,你需要掌握。
30 18
|
SQL 存储 Java
Java面向对象程序设计综合练习1(选择题)(下)
Java面向对象程序设计综合练习1(选择题)(下)
1188 0
Java面向对象程序设计综合练习1(选择题)(下)
|
Java Python
Java面向对象程序设计综合练习1(选择题)(上)
Java面向对象程序设计综合练习1(选择题)(上)
570 0
Java面向对象程序设计综合练习1(选择题)(上)
|
存储 物联网 Java
Java面向对象程序设计综合练习2(编程题)(下)
Java面向对象程序设计综合练习2(编程题)(下)
514 0
|
Java
Java面向对象程序设计综合练习2(编程题)(上)
Java面向对象程序设计综合练习2(编程题)(上)
479 0
|
JavaScript 安全 前端开发
【重温基础】4.函数
【重温基础】4.函数
116 0
青出于蓝-了不起的继承类 | 带你学《Java面向对象编程》之三十六
本节带领读者提出问题,引出疑惑后,提出了解决问题的方法-继承,为读者首次介绍了面向对象的第二大特征-继承性。
青出于蓝-了不起的继承类   | 带你学《Java面向对象编程》之三十六

热门文章

最新文章

下一篇
开通oss服务