0

0

C++模板方法模式如何设计 固定算法骨架与可变步骤实现

P粉602998670

P粉602998670

发布时间:2025-07-31 08:06:01

|

337人浏览过

|

来源于php中文网

原创

模板方法模式的核心是“骨架与细节分离”,它通过基类定义算法的固定流程,并将可变步骤推迟到子类实现。其关键在于使用抽象基类定义算法骨架,其中模板方法通常是非虚函数以防止被重写,而可变步骤则通过纯虚函数(强制子类实现)和虚函数(带默认实现,子类可选覆盖)来实现,同时非虚函数用于所有子类共享的固定步骤。此外,还引入钩子函数(hook method),为子类提供扩展点而不强制实现。例如在文档处理示例中,processdocument() 是模板方法,固定了“加载→解析→格式化→保存”的流程;loaddocument() 和 savedocument() 是非虚函数,表示通用步骤;parsecontent() 是纯虚函数,必须由子类实现;applyformatting() 是虚函数,带有默认实现;shouldlogprogress() 是钩子函数,允许子类选择性地启用某些行为。

C++模板方法模式如何设计 固定算法骨架与可变步骤实现

C++模板方法模式的核心在于它允许你定义一个算法的骨架,但将其中一些步骤的实现推迟到子类。说白了,就是把一个大任务的固定流程定死,但把其中一些个性化的、可变的部分留给不同的实现者去完成。

C++模板方法模式如何设计 固定算法骨架与可变步骤实现

要设计C++中的模板方法模式,你需要一个抽象基类来承载这个算法的骨架。这个基类会包含一个非虚的(或有时是虚的,但通常不推荐覆盖整个模板)“模板方法”,它按特定顺序调用一系列“基本操作”。这些基本操作有些是纯虚函数,强制子类去实现;有些是虚函数,带有默认实现,子类可以选择性地覆盖;还有些是具体的非虚函数,代表所有子类都共享的固定步骤。

例如,想象一个处理文档的流程:

立即学习C++免费学习笔记(深入)”;

纳米漫剧流水线
纳米漫剧流水线

360推出的国内首个工业级AI漫剧生产平台

下载
C++模板方法模式如何设计 固定算法骨架与可变步骤实现
#include <iostream>
#include <string>
#include <vector>

// 抽象基类:文档处理器
class DocumentProcessor {
public:
    // 模板方法:定义处理文档的固定算法骨架
    // 通常是非虚的,确保算法流程不被子类改变
    void processDocument() {
        std::cout << "--- 开始处理文档 ---" << std::endl;
        // 固定步骤1:加载文档
        loadDocument();
        // 可变步骤1:解析内容(强制子类实现)
        parseContent();
        // 可变步骤2:应用格式(子类可选实现或使用默认)
        applyFormatting();
        // 固定步骤2:保存文档
        saveDocument();
        std::cout << "--- 文档处理完成 ---" << std::endl;
    }

protected:
    // 基本操作1:加载文档 (固定步骤)
    void loadDocument() {
        std::cout << "通用操作: 正在加载文档..." << std::endl;
        // 实际加载逻辑...
    }

    // 基本操作2:解析内容 (纯虚函数,强制子类实现)
    virtual void parseContent() = 0;

    // 基本操作3:应用格式 (虚函数,带默认实现,子类可覆盖)
    virtual void applyFormatting() {
        std::cout << "通用操作: 正在应用默认格式..." << std::endl;
    }

    // 基本操作4:保存文档 (固定步骤)
    void saveDocument() {
        std::cout << "通用操作: 正在保存文档..." << std::endl;
        // 实际保存逻辑...
    }

    // 钩子函数 (Hook Method): 子类可以重写以在特定点插入代码,但不强制
    virtual bool shouldLogProgress() const {
        return false; // 默认不记录进度
    }
};

// 具体子类:PDF文档处理器
class PdfDocumentProcessor : public DocumentProcessor {
protected:
    void parseContent() override {
        std::cout << "PDF处理器: 正在解析PDF特定内容..." << std::endl;
        // PDF解析逻辑...
    }

    // 覆盖默认格式化,应用PDF特定格式
    void applyFormatting() override {
        std::cout << "PDF处理器: 正在应用PDF排版格式..." << std::endl;
    }

    bool shouldLogProgress() const override {
        return true; // PDF处理器选择记录进度
    }
};

// 具体子类:文本文档处理器
class TextDocumentProcessor : public DocumentProcessor {
protected:
    void parseContent() override {
        std::cout << "文本处理器: 正在解析纯文本内容..." << std::endl;
        // 文本解析逻辑...
    }
    // 不覆盖applyFormatting(), 使用基类的默认实现
};

/*
// 示例用法
int main() {
    PdfDocumentProcessor pdfProcessor;
    pdfProcessor.processDocument();

    std::cout << "\n";

    TextDocumentProcessor textProcessor;
    textProcessor.processDocument();

    return 0;
}
*/

在这个例子里,processDocument() 就是模板方法,它固定了文档处理的流程:加载 -> 解析 -> 格式化 -> 保存。loadDocument()saveDocument() 是所有子类都共享的固定步骤。parseContent() 是纯虚函数,强制子类必须提供自己的解析逻辑。而 applyFormatting() 是一个虚函数,提供了默认实现,子类可以选择是否覆盖。shouldLogProgress() 则是一个典型的钩子函数,子类可以决定是否启用某个可选行为。

模板方法模式的核心思想是什么?

模板方法模式的核心,我觉得,就是“骨架与细节分离”。它提供了一种机制,让你在基类中定义一个算法的步骤序列——也就是那个“骨架”——而将其中某些具体步骤的实现推迟到子类。这就像是定好了菜谱的主流程(比如先切菜,再炒,最后调味),但具体切什么菜、怎么炒、用什么调料,则由不同的厨师(子类)来发挥。

C++模板方法模式如何设计 固定算法骨架与可变步骤实现

它特别强调一个原则,有时候被称为“好莱坞原则”:“别打电话给我们,我们会打电话给你。”这意味着,基类(算法的骨架)掌握着控制权,它在适当的时候会“调用”子类提供的具体实现。这保证了算法的整体结构稳定不变,同时又允许局部细节的灵活变化。这和策略模式有点不同,策略模式是客户端选择一个完整的算法,而模板方法模式是基类定义了一个算法的结构,然后让子类去填充其中的某些部分。可以说,模板方法模式更侧重于算法的“内聚性”和“流程控制”。

模板方法模式在C++中如何实现?有哪些关键点?

在C++里实现模板方法模式,主要依赖于面向对象的三大支柱之一:继承。

关键点包括:

  • 抽象基类与继承: 你需要一个抽象基类(或至少是一个带有虚函数的基类)来定义模板方法和那些可变的基本操作。子类会继承这个基类,并实现或覆盖其中的虚函数。
  • 虚函数与纯虚函数: 这是实现“可变步骤”的关键。
    • 纯虚函数 (= 0): 用于那些必须由子类提供的具体实现。如果一个基类包含纯虚函数,它就变成了抽象类,不能直接实例化,只能通过其具体子类来使用。这强制了子类必须实现这些步骤。
    • 虚函数(带默认实现): 用于那些有通用默认行为,但子类也可以选择性地提供自己实现的步骤。这提供了灵活性,子类可以不关心这些步骤,也可以完全定制。
    • 非虚函数: 用于那些在所有子类中都保持不变的固定步骤。
  • 模板方法的非虚性(通常): 模板方法本身(

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
go语言 面向对象
go语言 面向对象

本专题整合了go语言面向对象相关内容,阅读专题下面的文章了解更多详细内容。

58

2025.09.05

java面向对象
java面向对象

本专题整合了java面向对象相关内容,阅读专题下面的文章了解更多详细内容。

63

2025.11.27

go语言 面向对象
go语言 面向对象

本专题整合了go语言面向对象相关内容,阅读专题下面的文章了解更多详细内容。

58

2025.09.05

java面向对象
java面向对象

本专题整合了java面向对象相关内容,阅读专题下面的文章了解更多详细内容。

63

2025.11.27

页面置换算法
页面置换算法

页面置换算法是操作系统中用来决定在内存中哪些页面应该被换出以便为新的页面提供空间的算法。本专题为大家提供页面置换算法的相关文章,大家可以免费体验。

494

2023.08.14

Go高并发任务调度与Goroutine池化实践
Go高并发任务调度与Goroutine池化实践

本专题围绕 Go 语言在高并发任务处理场景中的实践展开,系统讲解 Goroutine 调度模型、Channel 通信机制以及并发控制策略。内容包括任务队列设计、Goroutine 池化管理、资源限制控制以及并发任务的性能优化方法。通过实际案例演示,帮助开发者构建稳定高效的 Go 并发任务处理系统,提高系统在高负载环境下的处理能力与稳定性。

22

2026.03.10

Kotlin Android模块化架构与组件化开发实践
Kotlin Android模块化架构与组件化开发实践

本专题围绕 Kotlin 在 Android 应用开发中的架构实践展开,重点讲解模块化设计与组件化开发的实现思路。内容包括项目模块拆分策略、公共组件封装、依赖管理优化、路由通信机制以及大型项目的工程化管理方法。通过真实项目案例分析,帮助开发者构建结构清晰、易扩展且维护成本低的 Android 应用架构体系,提升团队协作效率与项目迭代速度。

48

2026.03.09

JavaScript浏览器渲染机制与前端性能优化实践
JavaScript浏览器渲染机制与前端性能优化实践

本专题围绕 JavaScript 在浏览器中的执行与渲染机制展开,系统讲解 DOM 构建、CSSOM 解析、重排与重绘原理,以及关键渲染路径优化方法。内容涵盖事件循环机制、异步任务调度、资源加载优化、代码拆分与懒加载等性能优化策略。通过真实前端项目案例,帮助开发者理解浏览器底层工作原理,并掌握提升网页加载速度与交互体验的实用技巧。

93

2026.03.06

Rust内存安全机制与所有权模型深度实践
Rust内存安全机制与所有权模型深度实践

本专题围绕 Rust 语言核心特性展开,深入讲解所有权机制、借用规则、生命周期管理以及智能指针等关键概念。通过系统级开发案例,分析内存安全保障原理与零成本抽象优势,并结合并发场景讲解 Send 与 Sync 特性实现机制。帮助开发者真正理解 Rust 的设计哲学,掌握在高性能与安全性并重场景中的工程实践能力。

216

2026.03.05

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Django 教程
Django 教程

共28课时 | 4.9万人学习

SciPy 教程
SciPy 教程

共10课时 | 1.9万人学习

Sass 教程
Sass 教程

共14课时 | 0.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号