当前位置: 首页 > backend >正文

C++抽象基类三重防线:纯虚函数与保护构造的深度实践

目录

抽象类

纯虚函数

只定义了protected构造函数的类


抽象类

抽象类有两种形式:

1 . 声明了纯虚函数的类,称为抽象类

2 . 只定义了protected型构造函数的类,也称为抽象类

纯虚函数

纯虚函数是一种特殊的虚函数,在许多情况下,在基类中不能对虚函数给出有意义的实现,而把它声明为纯虚函数,它的实现留给该基类的派生类去做。这就是纯虚函数的作用。纯虚函数的格式如下:

class 类名 {
public:virtual 返回类型 函数名(参数 ...) = 0;
};

在基类中声明纯虚函数就是在告诉派生类的设计者 —— 你必须提供一个纯虚函数的实现,但我不知道你会怎样实现它。

多个派生类可以对纯虚函数进行多种不同的实现,但是都需要遵循基类给出的接口(纯虚函数的声明)。

声明了纯虚函数的类成为抽象类,抽象类不能实例化对象。

看一个简单例子:

class A
{
public:virtual void print() = 0;virtual void display() = 0;
};class B
: public A
{
public:virtual void print() override{cout << "B::print()" << endl;}
};class C
: public B
{
public:virtual void display() override{cout << "C::display()" << endl;}
};void test0(){//A类定义了纯虚函数,A类是抽象类//抽象类无法创建对象//A a;//error//B b;//errorC c;A * pa2 = &c;pa2->print();pa2->display();
}

在A类中声明纯虚函数,A类就是抽象类,无法创建对象;

在B类中去覆盖A类的纯虚函数,如果把所有的纯虚函数都覆盖了(都实现了),B类可以创建对象;只要还有一个纯虚函数没有实现,B类也会是抽象类,也无法创建对象;

再往下派生C类,完成所有的纯虚函数的实现,C类才能够创建对象。

最顶层的基类(声明纯虚函数的类)虽然无法创建对象,但是可以定义此类型的指针,指向派生类对象,去调用实现好的纯虚函数。

—— 这种使用方式也归类为动态多态,尽管不符合第一个条件(基类中声明纯虚函数,而非定义),最终的效果仍然是基类指针调用到了派生类实现的虚函数,属于动态多态的特殊情况。

纯虚函数使用案例:

实现一个图形库,获取图形名称,获取图形之后计算它的面积

#include <iostream>
#include <math.h>
using namespace std;
class Figure
{
public:virtual string getName() const = 0;virtual double getArea() const = 0;};
void display(Figure& fig)
{cout << fig.getName()<< "的面积是:"<< fig.getArea() << endl;
}class Rectangle //矩形
:public Figure
{
public:Rectangle(double len, double wid):_len(len),_wid(wid){}string getName() const override{return "矩形";}double getArea() const override{return _len * _wid;}
private:double _len;double _wid;
};
class Circle 
:public Figure
{
public:Circle(double r):_r(r){}string getName() const override{return "圆形";}double getArea() const override{return  PI * pow(_r, 2);}
private://constexpr 将运行时常量转化为编译时常量//static constexpr double PI = 3.14;static const double PI;double _r;
};
const double Circle::PI = 3.14;class Triangle:public Figure
{
public:Triangle(double a, double b, double c):_a(a),_b(b),_c(c){}string getName() const override{return "三角形";}double getArea() const override{double p = (_a + _b + _c) / 2;return sqrt(p * (p - _a) * (p - _b) * (p - _c));}
private:double _a;double _b;double _c;
};
void test()
{Rectangle r(10, 20);Circle c(2);Triangle t(3, 4, 5);display(r);display(c);display(t);
}
int main()
{test();return 0;}

基类Figure中定义纯虚函数,交给多个派生类去实现,最后可以使用基类的指针(引用)指向(绑定)不同类型的派生类对象,再去调用已经被实现的纯虚函数。

纯虚函数就是为了后续扩展而预留的接口。

只定义了protected构造函数的类

如果一个类只定义了protected型的构造函数而没有提供public构造函数,无论是在外部还是在派生类中作为其对象成员都不能创建该类的对象,但可以由

其派生出新的类,这种能派生新类,却不能创建自己对象的类是另一种形式的抽象类。

Base类只定义了protected属性的构造函数,不能创建Base类的对象,但是可以定义Base类的指针—— Base类是抽象类

如果Derived类也只定义了protected属性的构造函数,Derived类也是抽象类,无法创建对象,但是可以定义指针指向派生类对象

那么还需要再往下派生,一直到某一层提供了public的构造函数,才能创建对象。

class Base {
protected:Base(int base): _base(base) { cout << "Base()" << endl; }int _base;
};
class Derived
: public Base {
public:Derived(int base, int derived): Base(base), _derived(derived){ cout << "Derived(int,int)" << endl; }void print() const{cout << "_base:" << _base<< ", _derived:" << _derived << endl;}
private:int _derived;
};void test()
{Base base(1);//errorDerived derived(1, 2);
}

 

解释如下:

基类的构造函数不能继承,只是可以通过创建基类子对象的形式来进行调用,而不能直接在派生类中直接调用基类的构造函数,所以不可以创建成员子对象。

将构造函数设置为私有,是不能进行继承的。

http://www.xdnf.cn/news/3801.html

相关文章:

  • js单调栈解题模板
  • skynet.socket.recv 没有处理分包问题
  • 办公文档全能处理工具功能解析
  • GR00t 安装使用教程踩坑记录
  • 专为焦油介质打造:煤焦油专用气动硬密封调节 V 型球阀(带手动)的卓越特点-耀圣
  • mvvm 如何 实现 MultiBinding 与转换器
  • SCAU18124--N皇后问题
  • 基于Vue2 + Element 实现任务列表管理功能的详细教程
  • tp5 php获取农历年月日干支甲午
  • MCP协议的使用分享
  • 数据库=====
  • 2025 年最新 Python 语言实现网易企业邮箱邮件推送验证码详细教程(更新中)
  • 智能决策支持系统的基本概念与理论体系
  • Ubuntu下安装Node.js
  • 【java八股文】深入浅出synchronized优化原理
  • 嵌入式Linux应用项目----智能网关
  • Docker Compose:服务编排:批量管理多个容器
  • 《Java高级编程:从原理到实战 - 进阶知识篇四》
  • 利用Elixir中的原子特性 + 错误消息泄露 -- Atom Bomb
  • 深度思考Qwen3
  • MySQL 中日期相减的完整指南
  • # 基于词袋模型(BoW)的猫狗图像分类实践
  • vue的diff算法是什么、比较方式,原理分析、示例解释讲解
  • 迭代器的思想和实现细节
  • 【序列化与反序列化详解】
  • 【漫话机器学习系列】237. TSS总平方和
  • 【2025软考高级架构师】——未来信息综合技术(11)
  • C++笔记-多态(包含虚函数,纯虚函数和虚函数表等)
  • 在MySQL中建索引时需要注意哪些事项?
  • Vue3源码学习5-不使用 `const enum` 的原因