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

设计模式每日硬核训练 Day 18:备忘录模式(Memento Pattern)完整讲解与实战应用

🔄 回顾 Day 17:中介者模式小结

在 Day 17 中,我们学习了中介者模式(Mediator Pattern):

  • 用一个中介者集中管理对象之间的通信。
  • 降低对象之间的耦合,适用于聊天系统、GUI 控件联动、塔台调度等。

今天进入一个非常贴近用户操作体验的设计模式——备忘录模式(Memento Pattern)

备忘录模式:在不破坏封装的前提下,保存对象的内部状态,便于后续恢复到某一状态。

它是“撤销 / 恢复”操作背后的设计思想核心。


一、备忘录模式的应用动机

在许多软件中,我们都可以看到“撤销(Undo)”、“恢复(Redo)”等功能:

  • 文本编辑器可以撤销几次输入
  • 游戏中可以回到某个存档
  • 图像处理工具支持操作历史回退

这就要求:

  • 程序能保存某个时刻的状态
  • 恢复时不依赖外部记录
  • 保证对象内部状态的私有性(封装)

✅ 所以我们引入备忘录模式:由备忘录(Memento)保存对象状态,管理员(Caretaker)持有备忘录,对象(Originator)可保存/恢复。


二、结构图(UML)

+----------------+
| Originator     |
+----------------+
| +createMemento |
| +restore(m)    |
+----------------+|v
+----------------+
| Memento        |
+----------------+
|  state         |
+----------------+^|
+----------------+
| Caretaker      |
+----------------+
|  history       |
+----------------+

在这里插入图片描述

三、角色说明

角色说明
Originator发起人:定义要保存的状态,并创建和恢复备忘录
Memento备忘录:存储发起人对象的状态,不提供修改接口
Caretaker管理者:保存备忘录对象,不操作其内容,仅作管理使用

四、C++ 实现:文本编辑器 Undo 示例

我们模拟一个文本编辑器,每次输入文本都可以保存当前状态。

✅ Originator 类:TextEditor

class Memento {std::string state_;
public:Memento(const std::string& s) : state_(s) {}std::string getState() const { return state_; }
};class TextEditor {std::string text_;
public:void type(const std::string& newText) {text_ += newText;}std::shared_ptr<Memento> save() {return std::make_shared<Memento>(text_);}void restore(std::shared_ptr<Memento> m) {text_ = m->getState();}void show() const {std::cout << "当前内容:" << text_ << std::endl;}
};

✅ Caretaker:备忘录栈

class Caretaker {std::stack<std::shared_ptr<Memento>> history_;
public:void backup(std::shared_ptr<Memento> m) {history_.push(m);}std::shared_ptr<Memento> undo() {if (!history_.empty()) {auto m = history_.top();history_.pop();return m;}return nullptr;}
};

✅ 使用示例

int main() {TextEditor editor;Caretaker caretaker;editor.type("Hello ");caretaker.backup(editor.save());editor.type("World!");caretaker.backup(editor.save());editor.type(" This should be undone.");editor.show();editor.restore(caretaker.undo());editor.show();editor.restore(caretaker.undo());editor.show();return 0;
}

输出:

当前内容:Hello World! This should be undone.
当前内容:Hello World!
当前内容:Hello

五、实际项目中的应用场景

场景应用说明
编辑器(文本、图像)操作历史,撤销恢复
游戏进度管理存档机制,一键恢复到特定状态
配置参数修改一键还原到默认参数或历史设置
工作流状态保存在流程推进过程中保存流程中间状态
数据库事务管理快照、事务回滚

六、优缺点分析

✅ 优点:

  • 保留对象状态,支持撤销与恢复
  • 不破坏封装性,状态由对象自身保存
  • 多个备份版本可管理

❗ 缺点:

  • 状态快照可能占用较大内存
  • 多次保存增加性能开销
  • 管理复杂,需注意备份何时保存/清理

七、与命令、原型模式对比

模式意图特点
备忘录 Memento保存对象内部状态封装状态,支持恢复
命令 Command将操作封装为对象,支持撤销记录操作动作,而非对象状态本身
原型 Prototype克隆对象一般用于新对象创建,非状态回滚

八、面试回答模板

“我们在图像处理系统中使用备忘录模式保存图像编辑的中间状态。每次用户进行滤镜、剪裁、调整操作时,会生成一个状态快照,存入备忘录栈。当用户点击撤销时,恢复到上一个状态。该方案确保封装性,同时支持多层撤销。”

✅ 强调:状态封装、用户体验、栈式回滚逻辑


九、口诀记忆

“封装状态不泄露,快照回滚不出错;历史栈中找快照,一键恢复没烦恼。”


十、明日预告:Day 19

解释器模式(Interpreter Pattern):为语言构建解释器,对语法规则建模,实现表达式的解析与执行。

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

相关文章:

  • 全球化电商平台AWS云架构设计
  • 矩阵置零(中等)
  • 设计模式-基础概念学习总结(继承、多态、虚方法、方法重写)
  • 深入理解块级格式化上下文(BFC)
  • 文本三剑客
  • 字符串匹配 之 拓展 KMP算法(Z算法)
  • 数据集-目标检测系列- 印度人脸 检测数据集 indian face >> DataBall
  • 深度解析:从 GPT-4o“谄媚”到 Deepseek“物理腔”,透视大模型行为模式的底层逻辑与挑战
  • Unity:AddTorque()(增加旋转力矩)
  • uniapp 云开发全集 云数据库
  • JavaScript 笔记 --- part7 --- JS进阶 (part2)
  • 【信息系统项目管理师-论文真题】2008上半年论文详解(包括解题思路和写作要点)
  • Python生活手册-NumPy数组创建:从快递分拣到智能家居的数据容器
  • 互联网大厂Java求职面试:AI大模型与云原生架构设计深度解析
  • 【学习心得】Xtuner模型qlora微调时错误记录
  • 【嘉立创EDA】FPCB(Flexible-PCB)柔性软板设计如何增加补强层
  • 反常积分(广义积分)
  • Redis总结(六)redis持久化
  • C++ 适配器模式详解
  • Java中使用Lock简化同步机制
  • 安装SDL和FFmpeg
  • 强化学习ppo算法在大语言模型上跑通
  • [ 设计模式 ] | 单例模式
  • Android学习总结之GetX库篇(场景运用)
  • 智能合约在去中心化金融(DeFi)中的核心地位与挑战
  • 机器学习中常见搜索算法
  • 代码随想录算法训练营第三十二天
  • Scrapy爬虫实战:如何用Rules实现高效数据采集
  • STM32教程:DMA运用及代码(基于STM32F103C8T6最小系统板标准库开发)*详细教程*
  • Vue3响应式原理那些事