C++如何实现适配器设计模式?C++结构型设计模式实例【代码重构】

发布时间 - 2025-12-25 00:00:00    点击率:
适配器模式通过封装+委托将不兼容接口转为期望接口,含类适配器(继承实现,零开销)和对象适配器(组合复用,更灵活),STL中stack/queue等即典型应用,适用于重构中安全过渡。

适配器模式(Adapter Pattern)在C++中用于让两个不兼容的接口协同工作——它不修改原有类,而是通过封装+委托,把“旧接口”转成“新期望接口”。核心就两点:目标接口(Target)、被适配者(Adaptee)、适配器(Adapter)作为中间层。

类适配器:继承 + 实现(静态绑定)

适用于 Adaptee 是具体类、且允许继承的场景。Adapter 同时继承 Target 抽象接口和 Adaptee 类,用继承复用 Adaptee 的实现。

示例:现有老日志类 OldLogger 只有 logString(),但新系统要求符合 ILogger 接口(含 write()level()):

class ILogger {
public:
    virtual void write(const std::string& msg) = 0;
    virtual int level() const = 0;
    virtual ~ILogger() = default;
};

class OldLogger { public: void logString(const std::string& s) { std::cout << "[OLD] " << s << "\n"; } };

class LoggerAdapter : public ILogger, private OldLogger { // 私有继承避免接口污染 public: void write(const std::string& msg) override { logString(msg); // 委托给旧实现 } int level() const override { return 2; } // 适配逻辑(比如固定为INFO级) };

✅ 优点:编译期绑定,零开销;可直接复用 Adaptee 的所有公有/保护成员。
❌ 注意:C++ 不支持多继承接口时若 Target 已是类而非抽象基类,需谨慎;私有继承更安全,避免外部误调 OldLogger 接口。

对象适配器:组合 + 指针(动态绑定,更常用)

Adapter 持有 Adaptee 的引用或智能指针,通过组合复用,解耦更强,也支持运行时切换被适配对象。

class LoggerAdapterObj : public ILogger {
private:
    std::shared_ptr adaptee_;
public:
    explicit LoggerAdapterObj(std::shared_ptr logger)
        : adaptee_(std::move(logger)) {}
void write(const std::string& msg) override {
    if (adaptee_) adaptee_->logString(msg);
}

int level() const override { return 3; }

};

✅ 更灵活:适配任意 OldLogger 实例(包括派生类);易于单元测试(可注入 mock);符合合成复用原则。
✅ 支持适配多个 Adaptee(如同时包装网络日志 + 文件日志)。
⚠️ 少量间接调用开销(通常可忽略)。

STL 中的天然适配器:stack、queue、priority_queue

C++ 标准库本身就是适配器模式的典范——它们不自己存数据,而是「包装」一个底层容器(如 deque、vector、list),只暴露受限接口:

  • std::stack 默认基于 std::deque,只开放 push()top()pop()
  • std::queue 同样包装 deque/list,只提供 FIFO 接口
  • std::priority_queue 包装 vector + make_heap 算法,提供堆语义

你甚至可以自定义:std::stack> —— 这就是把 vector “适配”成栈行为。无需重写存储逻辑,专注接口契约。

重构场景:当 legacy API 需对接新框架时

比如老项目用 int process(char* buf, int len) 处理数据,而新模块要求 std::string processData(const std::string&)

  • 别改老函数(可能被多处调用、无权限或风险高)
  • 写一个 Adapter 函数对象或轻量类,内部调用老函数并做字符串 ↔ char* 转换
  • 新模块只依赖 Adapter 接口,后续可替换成纯现代实现而不影响调用方

这样既保持向后兼容,又为逐步替换铺路——适配器是重构中“安全过渡”的关键胶水。

基本上就这些。类适配器适合简单、确定的继承关系;对象适配器更通用、更推荐;而理解 STL 容器适配器,能帮你写出更自然的 C++ 接口。不复杂但容易忽略——关键是守住“不改旧、不强求新、只做翻译”这个边界。


#   # c++  # 标准库  # red  # String  # 封装  # const  # 字符串  # char  # int  # 指针  # 继承  # 私有继承  # 多继承  # 接口  #   # 委托  # len  # 对象  # 算法  # 重构  # 复用  # 绑定  # 适用于  # 更灵活  # 中间层  # 多个  # 这就是  # 帮你  # 而不 


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


相关推荐: 如何在IIS中新建站点并解决端口绑定冲突?  利用python获取某年中每个月的第一天和最后一天  高性能网站服务器配置指南:安全稳定与高效建站核心方案  Angular 表单中正确绑定输入值以确保提交与验证正常工作  如何在阿里云ECS服务器部署织梦CMS网站?  Linux虚拟化技术教程_KVMQEMU虚拟机安装与调优  用v-html解决Vue.js渲染中html标签不被解析的问题  Internet Explorer官网直接进入 IE浏览器在线体验版网址  历史网站制作软件,华为如何找回被删除的网站?  laravel怎么配置Redis作为缓存驱动_laravel Redis缓存配置教程  Laravel如何使用.env文件管理环境变量?(最佳实践)  Claude怎样写结构化提示词_Claude结构化提示词写法【教程】  Laravel的HTTP客户端怎么用_Laravel HTTP Client发起API请求教程  Laravel如何配置中间件Middleware_Laravel自定义中间件拦截请求与权限校验【步骤】  新三国志曹操传主线渭水交兵攻略  个人网站制作流程图片大全,个人网站如何注销?  手机怎么制作网站教程步骤,手机怎么做自己的网页链接?  如何在云主机快速搭建网站站点?  Laravel如何使用Guzzle调用外部接口_Laravel发起HTTP请求与JSON数据解析【详解】  Win11怎么关闭透明效果_Windows11辅助功能视觉效果设置  Android 常见的图片加载框架详细介绍  大学网站设计制作软件有哪些,如何将网站制作成自己app?  如何快速搭建二级域名独立网站?  Laravel如何实现全文搜索_Laravel Scout集成Algolia或Meilisearch教程  BootStrap整体框架之基础布局组件  详解jQuery停止动画——stop()方法的使用  Laravel怎么使用Collection集合方法_Laravel数组操作高级函数pluck与map【手册】  JS中对数组元素进行增删改移的方法总结  在centOS 7安装mysql 5.7的详细教程  想要更高端的建设网站,这些原则一定要坚持!  如何在阿里云域名上完成建站全流程?  Python并发异常传播_错误处理解析【教程】  实现点击下箭头变上箭头来回切换的两种方法【推荐】  零基础网站服务器架设实战:轻量应用与域名解析配置指南  Laravel如何使用模型观察者?(Observer代码示例)  Laravel如何自定义错误页面(404, 500)?(代码示例)  如何快速生成高效建站系统源代码?  Laravel如何创建和注册中间件_Laravel中间件编写与应用流程  Laravel如何发送系统通知?(Notification渠道示例)  悟空识字如何进行跟读录音_悟空识字开启麦克风权限与录音  Laravel怎么配置.env环境变量_Laravel生产环境敏感数据保护与读取【方法】  厦门模型网站设计制作公司,厦门航空飞机模型掉色怎么办?  HTML透明颜色代码在Angular里怎么设置_Angular透明颜色使用指南【详解】  如何快速查询网址的建站时间与历史轨迹?  惠州网站建设制作推广,惠州市华视达文化传媒有限公司怎么样?  再谈Python中的字符串与字符编码(推荐)  如何用已有域名快速搭建网站?  如何自定义safari浏览器工具栏?个性化设置safari浏览器界面教程【技巧】  WEB开发之注册页面验证码倒计时代码的实现  轻松掌握MySQL函数中的last_insert_id()