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

【读书笔记】《C++ Software Design》第七章:Bridge、Prototype 与 External Polymorphism

《C++ Software Design》第七章:Bridge、Prototype 与 External Polymorphism

在追求低耦合、高扩展和良好封装的 C++ 软件架构设计中,如何抽象行为、复用结构、隔离模块依赖是关键。《C++ Software Design》第七章聚焦三个模式——BridgePrototypeExternal Polymorphism,它们分别解决“实现-接口解耦”、“抽象复制”和“非侵入式运行时多态”三大类问题。


Guideline 28:用 Bridge 模式解除物理依赖

什么是 Bridge 模式?

Bridge 模式的核心思想是将“抽象接口(Abstraction)”与“具体实现(Implementation)”分离,从而使它们可以独立演化

目的:将接口和实现分离以减少物理依赖、降低构建成本。

UML 示意:
[Abstraction] → [Implementation Interface] ← [ConcreteImplementation]

一个动机示例

假设我们有多个图形类 Shape(如 Circle, Rectangle),它们依赖不同的渲染器(OpenGL、Direct2D):

class OpenGLRenderer {
public:void drawCircle() { std::cout << "OpenGL Circle\n"; }
};class Circle {OpenGLRenderer* renderer;
public:void draw() {renderer->drawCircle();  // 强耦合!}
};

这使得 Circle 绑定了 OpenGLRenderer,难以扩展或替换。

使用 Bridge 解耦

class Renderer {
public:virtual void drawCircle() = 0;
};class OpenGLRenderer : public Renderer {
public:void drawCircle() override {std::cout << "OpenGL: Circle\n";}
};class Shape {
protected:Renderer* renderer;
public:Shape(Renderer* r) : renderer(r) {}virtual void draw() = 0;
};class Circle : public Shape {
public:using Shape::Shape;void draw() override {renderer->drawCircle();}
};

这样,ShapeRenderer 的继承树可独立变化,真正解耦。

Bridge 与 Strategy 的对比

维度BridgeStrategy
抽象目的解耦接口与实现封装算法的变体
运行时行为固定组件间协作结构可动态替换策略
封装粒度架构级别(模块解耦)算法级别(行为切换)

Pimpl(Pointer to Implementation)

class Widget {struct Impl;std::unique_ptr<Impl> pImpl;
public:Widget();~Widget();void draw();
};

优点:

  • 隐藏实现细节(API 稳定)
  • 减少头文件依赖
  • 减少编译时间

Bridge 的短板分析

  • 增加间接层次,导致理解和调试复杂性上升
  • 对小型系统或临时原型而言设计成本较高
  • 若实现接口变化频繁,也会影响稳定性

Guideline 29:桥接设计的性能权衡

虽然 Bridge 模式提升了解耦能力,但它也带来了性能成本,特别是在高频调用路径中。

性能影响分析

  • 虚函数调用开销
  • 缓存局部性降低
  • 过度抽象使得优化受限

优化手段:Partial Bridge

将不变部分内联进抽象类,仅对可变部分桥接:

class Renderer {
public:virtual void drawCircle(float radius) = 0;
};class Circle {Renderer* renderer;float radius;
public:void draw() {if (radius > 10) {// Fast path: inline draw logic} else {renderer->drawCircle(radius); // Bridge}}
};

Guideline 30:使用 Prototype 实现抽象复制

Prototype 模式是什么?

当你需要复制未知类型的对象时,不能直接使用复制构造函数或赋值。这时 Prototype 提供了解耦复制的方法。

class Animal {
public:virtual std::unique_ptr<Animal> clone() const = 0;virtual void sound() const = 0;
};class Sheep : public Animal {
public:std::unique_ptr<Animal> clone() const override {return std::make_unique<Sheep>(*this);}void sound() const override {std::cout << "Baaa\n";}
};

与 std::variant 的对比

模式Prototypestd::variant
类型行为基于继承和 clone静态多态(编译时分发)
动态性高(运行时识别)低(编译期已知)
类型安全差(需手动类型管理)高(类型封装安全)

Prototype 缺点分析

  • 所有子类需实现 clone(),违背开闭原则
  • 容易遗漏深拷贝细节,产生 bug
  • 缺乏语言级支持,机制易错

Guideline 31:External Polymorphism(外部多态)

什么是 External Polymorphism?

传统多态依赖继承与虚函数,但这要求你侵入类定义。External Polymorphism 提供了一种非侵入式运行时多态实现方式。

示例:绘图对象的非侵入式调用

struct Circle {void draw() const {std::cout << "Draw Circle\n";}
};struct Square {void draw() const {std::cout << "Draw Square\n";}
};class Shape {struct Concept {virtual void draw() const = 0;virtual std::unique_ptr<Concept> clone() const = 0;virtual ~Concept() = default;};template<typename T>struct Model : Concept {T data;Model(const T& d) : data(d) {}void draw() const override { data.draw(); }std::unique_ptr<Concept> clone() const override {return std::make_unique<Model<T>>(*this);}};std::unique_ptr<Concept> self;
public:template<typename T>Shape(T obj) : self(std::make_unique<Model<T>>(obj)) {}void draw() const { self->draw(); }
};

特性分析

  • 不依赖虚函数或继承结构
  • 支持任意用户类型,接口透明
  • 可实现“按值”传递的运行时多态

与 Adapter 模式对比

特性External PolymorphismAdapter
对象结构无需继承或接口通常需目标接口/抽象类
灵活性高(支持任意类型)适用于已知接口转换
封装方式封装在中间结构体(Model)中封装在 Adapter 类中

缺点分析

  • 实现繁琐(需要手动写 Concept/Model)
  • 性能开销相较虚函数更大
  • 对于移动语义支持较难实现完全泛化

总结对比

模式核心作用优点缺点
Bridge抽象接口与实现分离架构解耦、支持独立演化抽象复杂、增加间接层级
Prototype抽象复制机制支持克隆任意派生对象需手动实现 clone,易出错
External Polym.非侵入运行时多态高灵活性、非侵入性类型擦除实现复杂,运行效率稍低

结语

Bridge、Prototype 和 External Polymorphism——分别应对模块解耦、对象复制和多态封装问题,是构建高可维护性、高可扩展性系统架构的有效手段。理解并掌握这些模式,不仅能提升你的 C++ 设计能力,也能让你更好地驾驭大型系统演化的复杂性。

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

相关文章:

  • IPC框架
  • [2025CVPR]GNN-ViTCap:用于病理图像分类与描述模型
  • 晋升指南-笔记
  • 【Docker基础】Dockerfile指令速览:环境与元数据指令详解
  • React强大且灵活hooks库——ahooks入门实践之状态管理类hook(state)详解
  • 【C++】多线程同步三剑客介绍
  • AutoLabor-ROS-Python 学习记录——第一章 ROS概述与环境搭建
  • leetGPU解题笔记(1)
  • STM32-第六节-TIM定时器-2(输出比较)
  • 【芯片笔记】ADF4159
  • 【论文阅读】AdaptThink: Reasoning Models Can Learn When to Think
  • 【Java Stream】基本用法学习
  • sql初学见解
  • 2025上海市“星光计划“信息安全管理与评估赛项二三阶段任务书
  • Spring高级特性——反射和动态代理的性能优化
  • Python---上下文管理器
  • 移动端设备本地部署大语言模型(LLM)
  • 无需付费即可利用AI消除音频噪声和生成字幕
  • 浏览器渲染原理与性能优化全解析
  • 【零基础入门unity游戏开发——unity3D篇】3D光源之——unity反射和反射探针技术
  • 在线事务处理OLTP(Online Transaction Processing)负载是什么?
  • 08.如何正确关闭文件
  • QML 自定义Model基础之QAbstractListModel
  • iw 命令 -- linux 无线管理
  • python kivy 打包apk
  • Ampace厦门新能安科技Verify 测评演绎数字推理及四色测评考点分析、SHL真题题库
  • 入职华为od一个月的感受
  • 用 Node.js 构建模块化的 CLI 脚手架工具,从 GitHub 下载远程模板
  • 【Vue】浏览器缓存 sessionStorage、localStorage、Cookie
  • 初级网安作业笔记1