C++ 怎么实现接口 C++纯虚函数与抽象类定义规范【架构】

发布时间 - 2026-01-24 00:00:00    点击率:
抽象类必须至少有一个纯虚函数(virtual void f() = 0),否则仍可实例化;接口类应零数据成员、全纯虚、公有虚析构,且职责清晰。

抽象类必须至少有一个 virtual 函数且含 = 0

不是所有带 virtual 的类都是抽象类,只有声明了纯虚函数(即形如 virtual void foo() = 0;)的类才不可实例化。编译器会直接报错:error: cannot declare variable 'x' to be of abstract type 'A'

常见错误是忘记写 = 0,只写 virtual void foo(); —— 这只是虚函数,不是纯虚,类仍可实例化,无法起到接口约束作用。

  • = 0 必须紧贴函数声明末尾,不能有空格隔开(virtual void f()=0; 合法,但 virtual void f() =0; 在部分老编译器可能报错)
  • 纯虚函数可以有函数体(定义在类外),用于提供默认实现,子类通过 A::f() 显式调用
  • 析构函数建议也声明为 virtual(哪怕纯虚),否则通过基类指针 delete 派生类对象时行为未定义

override 不是可选修饰,而是强制类型安全的必要手段

子类重写纯虚函数时,不加 override 不报错,但极易因签名微小差异(如参数 const 修饰、引用类型、noexcept)导致“看似重写实则新增”——此时对象仍为抽象类,无法实例化,且错误常延迟到链接或运行时才暴露。

例如:virtual void draw() const = 0; 在基类中声明,子类写 void draw() { ... }(漏了 const),没加 override 就不会报错,但该子类仍是抽象类。

  • 所有重写函数都应显式标注 override,由编译器校验签名一致性
  • 基类虚函数若不希望被进一步重写,可用 final(如 virtual void f() final;
  • 使用 -Woverloaded-virtual-Wsuggest-override 编译选项可辅助发现遗漏

接口类(Interface Class)应满足零数据成员 + 全纯虚 + 公共非虚析构

真正意义上的 C++ 接口(类似 Java interface)需严格规避实现细节:不能有非静态数据成员、不能有构造函数实现、不能有 protected 成员(除非有意暴露内部钩子)。

典型误用是把“工具函数”塞进接口类,比如加一个 std::string get_name() const { return name_; } —— 这立刻让类脱离接口定位,变成半抽象基类,破坏依赖倒置原则。

  • 接口类的析构函数应为 public: virtual ~Interface() = default;= 0(推荐前者,避免强制子类实现空析构)
  • 若需默认行为,应通过组合而非继承:定义纯接口类 IReader,另写一个 DefaultReader 实现它,供其他类组合使用
  • 多个接口应按职责拆分(IReadable / IWritable),避免大而全的“上帝接口”

多继承接口时,虚继承不是默认选项,但菱形继承必须处理

当两个接口类(如 IStreamableIJsonSerializable)都继承自同一根接口(如 IObject),而具体类同时实现二者时,若未对 IObject 使用 virtual 继承,会导致二义

性:obj->IObject::id() 无法解析。

这不是规范强制要求,而是实际工程中菱形继承出现频率远高于预期(尤其跨模块协作时),不提前预防会在后期引发难以调试的 ODR 或切片问题。

  • 只要存在公共祖先接口,就应在继承时写 : virtual public IObject
  • 虚继承会略微增加对象大小(vptr 开销)和访问成本,但接口类本身无数据成员,影响极小
  • static_assert(std::is_base_of_v, "Missing virtual inheritance"); 可在编译期兜底
接口真正的复杂点不在语法,而在职责边界是否清晰;一个标着 interface 注释却塞了三个私有缓存成员的类,比语法错误更难维护。


# java  # js  # json  # 工具  # c++  # stream 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: Laravel如何与Docker(Sail)协同开发?(环境搭建教程)  Laravel路由怎么定义_Laravel核心路由系统完全入门指南  JS去除重复并统计数量的实现方法  利用JavaScript实现拖拽改变元素大小  Laravel Docker环境搭建教程_Laravel Sail使用指南  简单实现jsp分页  如何快速搭建FTP站点实现文件共享?  教学论文网站制作软件有哪些,写论文用什么软件 ?  UC浏览器如何切换小说阅读源_UC浏览器阅读源切换【方法】  如何在Ubuntu系统下快速搭建WordPress个人网站?  网站页面设计需要考虑到这些问题  Laravel如何使用Telescope进行调试?(安装和使用教程)  UC浏览器如何设置启动页 UC浏览器启动页设置方法  个人摄影网站制作流程,摄影爱好者都去什么网站?  Android利用动画实现背景逐渐变暗  如何在 Python 中将列表项按字母顺序编号(a.、b.、c. …)  Laravel如何集成微信支付SDK_Laravel使用yansongda-pay实现扫码支付【实战】  Laravel如何使用Socialite实现第三方登录?(微信/GitHub示例)  Android中Textview和图片同行显示(文字超出用省略号,图片自动靠右边)  免费网站制作appp,免费制作app哪个平台好?  HTML 中如何正确使用模板变量为元素的 name 属性赋值  Linux系统运维自动化项目教程_Ansible批量管理实战  详解vue.js组件化开发实践  Win11怎么修改DNS服务器 Win11设置DNS加速网络【指南】  北京网站制作费用多少,建立一个公司网站的费用.有哪些部分,分别要多少钱?  购物网站制作费用多少,开办网上购物网站,需要办理哪些手续?  Python文件流缓冲机制_IO性能解析【教程】  在线制作视频的网站有哪些,电脑如何制作视频短片?  Laravel怎么创建控制器Controller_Laravel路由绑定与控制器逻辑编写【指南】  公司门户网站制作公司有哪些,怎样使用wordpress制作一个企业网站?  东莞专业网站制作公司有哪些,东莞招聘网站哪个好?  如何在香港免费服务器上快速搭建网站?  如何在阿里云虚拟主机上快速搭建个人网站?  高性价比服务器租赁——企业级配置与24小时运维服务  微信小程序 配置文件详细介绍  如何在橙子建站中快速调整背景颜色?  新三国志曹操传主线渭水交兵攻略  高端建站如何打造兼具美学与转化的品牌官网?  家族网站制作贴纸教程视频,用豆子做粘帖画怎么制作?  如何在阿里云部署织梦网站?  Laravel如何实现用户密码重置功能?(完整流程代码)  北京网站制作的公司有哪些,北京白云观官方网站?  Laravel如何实现API资源集合?(Resource Collection教程)  如何在腾讯云服务器上快速搭建个人网站?  香港服务器网站卡顿?如何解决网络延迟与负载问题?  如何在新浪SAE免费搭建个人博客?  Laravel如何优雅地处理服务层_在Laravel中使用Service层和Repository层  合肥制作网站的公司有哪些,合肥聚美网络科技有限公司介绍?  如何在阿里云完成域名注册与建站?  如何用PHP工具快速搭建高效网站?