在c++++中,抽象类通过包含至少一个纯虚函数定义。1. 使用virtual void func() = 0语法声明纯虚函数;2. 建议添加虚析构函数以确保正确析构;3. 抽象类不可实例化,只能作为基类;4. 派生类未实现所有纯虚函数则自身也成抽象类;5. 纯虚函数可提供实现供派生类调用;6. 设计接口时应保持简洁、命名明确;7. 考虑默认实现以减少重复代码;8. 根据是否需要共享状态或逻辑选择抽象类或接口类。

在C++中,抽象类的核心在于包含至少一个纯虚函数。纯虚函数没有实现,强制派生类去实现它。定义抽象类的关键是设计清晰、可扩展的接口,同时避免过度设计。

抽象类的基本定义方式
C++中通过将虚函数设为“纯虚函数”来定义抽象类。语法是在声明虚函数时赋值为0:
class Shape {
public:
virtual void draw() = 0; // 纯虚函数
virtual ~Shape() {} // 建议加上虚析构函数
};只要有至少一个纯虚函数,这个类就无法实例化,只能作为基类使用。这种结构非常适合定义接口或抽象行为。
立即学习“C++免费学习笔记(深入)”;

- 如果派生类不实现所有纯虚函数,它自己也会变成抽象类。
- 抽象类可以有普通成员函数、数据成员,不只是接口。
纯虚函数的作用与注意事项
纯虚函数的主要作用是定义一个必须被覆盖的接口。它让子类遵循统一的行为规范。
但要注意以下几点:

- 不要滥用纯虚函数:如果某个方法在大多数子类中实现相似,不如提供默认实现。
- 虚析构函数很重要:如果不定义虚析构函数,通过基类指针删除对象会导致未定义行为。
- 可以有实现的纯虚函数:虽然少见,但你可以为纯虚函数提供实现,供派生类调用。
例如:
class Base {
public:
virtual void foo() = 0;
};
void Base::foo() { /* 可以被派生类调用 */ }
class Derived : public Base {
public:
void foo() override {
Base::foo(); // 调用基类实现
}
};接口设计的最佳实践
抽象类常用于模拟“接口”,但在设计时应遵循一些实用原则:
- 保持接口简洁:只暴露必要的方法,减少耦合。
-
命名要明确一致:比如
draw()比render()更直观,或者根据上下文统一风格。 - 优先使用组合而非继承:不是所有问题都适合用抽象类解决,有时候策略模式或组合更灵活。
- 考虑默认实现:如果多个子类逻辑类似,可以考虑在抽象类中提供默认实现。
举个例子,如果你有一个图形库,可能一开始这样设计:
class Shape {
public:
virtual void draw() = 0;
virtual ~Shape() {}
};后续发现很多形状都需要填充颜色,那么可以加一个虚函数:
virtual void setFillColor(Color c);
但如果多数子类处理方式相同,可以直接给出默认实现,而不是强求每个子类重写。
抽象类 vs 接口类:如何选择?
严格来说,C++没有专门的“接口类”关键字(像Java的interface),但可以通过只含纯虚函数的类来模拟。
什么时候该用抽象类?什么时候该用全纯虚的“接口类”?
- 用抽象类:当你需要共享部分实现、有状态(成员变量)或构造/析构逻辑。
- 用接口类:当你只需要定义行为规范,不关心具体实现,希望实现类自由组合。
接口类一般长这样:
class IRenderable {
public:
virtual void render() = 0;
virtual ~IRenderable() {}
};这样的类便于实现多继承和插件式架构。
基本上就这些。抽象类的设计关键是围绕“行为抽象”展开,合理使用纯虚函数和默认实现,才能写出清晰、易维护的代码。










