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

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);
}

 

解释如下:

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

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

相关文章:

  • springAop代理责任链模式源码解析
  • 《解锁GCC版本升级:开启编程新世界大门》
  • Python蓝桥杯真题代码
  • 工作记录 2015-06-01
  • 数据库介绍以及windows下mysql安装
  • vector和string的迭代器
  • BG开发者日志505:项目总体情况
  • PowerPC架构详解:定义、应用及特点
  • 软件管理(安装方式)
  • MCP 探索:MCP 集成的相关网站 Smithery、PulseMCP 等
  • MySQL安装完全指南:从零开始到配置优化(附避坑指南)
  • 【Python生成器与迭代器】核心原理与实战应用
  • 【Python实战】飞机大战
  • 吾爱出品 [Windows] EndNote 21.5.18513 汉化补丁
  • 【进阶】C# 委托(Delegate)知识点总结归纳
  • Rotary Positional Embedding
  • QT6 源(72):阅读与注释单选框这个类型的按钮 QRadioButton,及各种属性验证,
  • 存在重复元素II(简单)
  • Three.js在vue中的使用(二)-加载、控制
  • web应用开发说明文档
  • 释新闻|新加坡大选今日投票:除了黄循财首次挂帅,还有哪些看点
  • 安徽六安特色产品将“组团”入沪,借力五五购物节开拓市场
  • 神十九都带回了哪些实验样品?果蝇等生命类样品已交付科学家
  • 揭秘神舟十九号返回舱“软着陆”关键:4台发动机10毫秒内同时点火
  • 专访|200余起诉讼,特朗普上台100天,美国已进入宪政危机
  • 中信银行一季度净利195.09亿增1.66%,不良率持平