C++ 基础

C++ 主页 C++ 概述 C++ 环境设置 C++ 基本语法 C++ 注释 C++ Hello World C++ 省略命名空间 C++ 标记 C++ 常量/字面量 C++ 关键字 C++ 标识符 C++ 数据类型 C++ 数字数据类型 C++ 字符数据类型 C++ 布尔数据类型 C++ 变量类型 C++ 变量作用域 C++ 多变量 C++ 基本输入/输出 C++ 修饰符类型 C++ 存储类 C++ 数字 C++ 枚举 C++ 枚举类 C++ 引用 C++ 日期和时间

C++ 运算符

C++ 运算符 C++ 算术运算符 C++ 关系运算符 C++ 逻辑运算符 C++ 位运算符 C++ 赋值运算符 C++ sizeof 运算符 C++ 条件运算符 C++ 逗号运算符 C++ 成员运算符 C++ 强制类型转换运算符 C++ 指针运算符 C++ 运算符优先级 C++ 一元运算符

C++ 控制语句

C++ 决策语句 C++ if 语句 C++ if else 语句 C++ 嵌套 if 语句 C++ switch 语句 C++ 嵌套 switch语句 C++ 循环类型 C++ while 循环 C++ for 循环 C++ do while 循环 C++ Foreach 循环 C++ 嵌套循环 C++ break 语句 C++ continue 语句 C++ goto 语句

C++ 字符串

C++ 字符串 C++ 循环遍历字符串 C++ 字符串长度 C++ 字符串连接 C++ 字符串比较

C++ 函数

C++ 函数 C++ 多函数参数 C++ 递归函数 C++ 返回值 C++ 函数重载 C++ 函数重写 C++ 默认参数

C++ 数组

C++ 数组 C++ 多维数组 C++ 指向数组的指针 C++ 将数组传递给函数 C++ 从函数返回数组

C++ 结构 &联合

C++ 结构 C++ 联合

C++ 指针

C++ 指针 C++ 解引用 C++ 修改指针

C++ 类和对象

C++ 面向对象 C++ 类 &对象 C++ 类成员函数 C++ 类访问修饰符 C++ 静态类成员 C++ 静态数据成员 C++ 静态成员函数 C++ 内联函数 C++ this 指针 C++ 友元函数 C++ 指向类的指针

C++ 构造函数

C++ 构造函数 &析构函数 C++ 默认构造函数 C++ 参数化构造函数 C++ 复制构造函数 C++ 构造函数重载 C++ 带默认参数的构造函数 C++ 委托构造函数 C++ 构造函数初始化列表 C++ 使用构造函数动态初始化

C++ 继承

C++ 继承 C++ 多重继承 C++ 多级继承

C++ 面向对象

C++ 重载 C++ 多态性 C++ 抽象 C++ 封装 C++ 接口 C++ 虚函数 C++ 纯虚函数与抽象类

C++ 文件处理

C++ 文件和流 C++ 文件读取

C++ 进阶

C++ 异常处理 C++ 动态内存 C++ 命名空间 C++ 模板 C++ 预处理器 C++ 信号处理 C++ 多线程 C++ Web 编程 C++ 套接字编程 C++ 并发 C++ 高级概念 C++ Lambda 表达式 C++ unordered_multiset

C++ 实用资源

C++ 问答 C++ 快速指南 C++ 速查表 C++ STL 教程 C++ 标准库 C++ 实用资源 C++ 讨论


C++ 中的接口(抽象类)

接口描述 C++ 类的行为或功能,但不指定该类的具体实现。

C++ 接口使用抽象类实现,这些抽象类不应与数据抽象混淆,数据抽象是将实现细节与关联数据分离的概念。

通过将类中的至少一个函数声明为纯虚函数,类就变成了抽象类。纯虚函数的声明方式是:在类声明中加上"= 0",如下所示:

class Box {
   public:
      // 纯虚函数
      virtual double getVolume() = 0;
      
   private:
      double length;      // Box 的长度
      double breadth;     // Box 的宽度
      double height;      // Box 的高度
};

抽象类(通常称为 ABC)的目的是提供一个合适的基类,以便其他类可以继承。抽象类不能用于实例化对象,而只能用作接口。尝试实例化抽象类的对象会导致编译错误。

因此,如果需要实例化 ABC 的子类,它必须实现每个虚函数,这意味着它支持 ABC 声明的接口。如果在派生类中未重写纯虚函数,然后尝试实例化该类的对象,则会导致编译错误。

可用于实例化对象的类称为具体类

抽象类示例

请考虑以下示例,其中父类为基类提供了一个接口,以实现名为 getArea() 的函数 -

#include <iostream>

using namespace std;

// 基类
class Shape {
   public:
      // 提供接口框架的纯虚函数。
      virtual int getArea() = 0;
      void setWidth(int w) {
         width = w;
      }
   
      void setHeight(int h) {
         height = h;
      }
   
   protected:
      int width;
      int height;
};
 
// 派生类
class Rectangle: public Shape {
   public:
      int getArea() { 
         return (width * height); 
      }
};

class Triangle: public Shape {
   public:
      int getArea() { 
         return (width * height)/2; 
      }
};
 
int main(void) {
   Rectangle Rect;
   Triangle  Tri;
 
   Rect.setWidth(5);
   Rect.setHeight(7);
   
   // 打印对象的面积。
   cout << "Total Rectangle area: " << Rect.getArea() << endl;

   Tri.setWidth(5);
   Tri.setHeight(7);
   
   // 打印对象的面积。
   cout << "Total Triangle area: " << Tri.getArea() << endl; 

   return 0;
}

当编译并执行上述代码时,它会产生以下结果 -

Total Rectangle area: 35
Total Triangle area: 17

您可以看到,一个抽象类如何根据 getArea() 定义一个接口,而另外两个类如何实现相同的函数,但使用不同的算法来计算特定形状的面积。

设计策略

面向对象系统可能会使用一个抽象基类来提供适用于所有外部应用程序的通用标准化接口。然后,通过从该抽象基类继承,可以形成具有类似操作的派生类。

外部应用程序提供的功能(即公共函数)在抽象基类中以纯虚函数的形式提供。这些纯虚函数的实现在与应用程序特定类型对应的派生类中提供。

即使在系统定义之后,这种架构也允许轻松地将新的应用程序添加到系统中。