C++备忘录模式:优雅实现对象状态保存与恢复

C++备忘录模式:优雅实现对象状态保存与恢复

C++备忘录模式:优雅实现对象状态保存与恢复

引言

在软件开发中,我们经常需要实现撤销操作、历史记录或状态回滚等功能。备忘录模式(Memento Pattern)正是为解决这类问题而生的设计模式。本文将深入探讨备忘录模式在C++中的实现与应用,帮助开发者掌握这一强大的设计工具。

备忘录模式概述

备忘录模式是一种行为设计模式,它允许在不破坏封装性的前提下捕获并外部化一个对象的内部状态,以便以后可以将该对象恢复到原先保存的状态【1†source】。该模式特别适合需要实现撤销操作、历史记录或快照功能的场景【1†source】【2†source】。

核心角色解析

1. Originator(发起人)

Originator是需要保存状态的对象,它:

  • 负责创建备忘录并记录当前状态
  • 可以使用备忘录恢复之前的状态
  • 知道如何正确保存和恢复自身状态【1†source】【2†source】

2. Memento(备忘录)

Memento是存储Originator状态的对象,特点包括:

  • 存储Originator的内部状态
  • 提供两个接口:宽接口(Originator使用)和窄接口(其他对象使用)【1†source】
  • 确保状态的封装性不被破坏

3. Caretaker(管理者)

Caretaker负责管理备忘录,其职责是:

  • 保存备忘录对象
  • 不操作备忘录内容
  • 只能通过窄接口与备忘录交互【1†source】

设计原则体现

备忘录模式很好地体现了多个面向对象设计原则:

  1. 单一职责原则:将状态保存的职责从Originator中分离出来【3†source】【7†source】,使Originator可以专注于核心业务逻辑【3†source】。
  2. 开闭原则:不需要修改Originator类就能扩展状态保存功能【3†source】,通过新增备忘录类实现这一扩展【3†source】。
  3. 迪米特法则:Caretaker不直接操作Memento内部细节【3†source】【6†source】,有效降低了系统耦合度【6†source】。

C++实现示例

#include<iostream>#include<string>#include<vector>// 备忘录类classMemento{private: std::string state;// 只有Originator可以访问私有成员friendclassOriginator;Memento(const std::string& s):state(s){} std::string GetState()const{return state;}};// 发起人类classOriginator{private: std::string state;public:voidSetState(const std::string& s){ std::cout <<"设置状态: "<< s << std::endl; state = s;} Memento*CreateMemento(){returnnewMemento(state);}voidRestoreFromMemento(const Memento* m){ state = m->GetState(); std::cout <<"恢复状态: "<< state << std::endl;}};// 管理者类classCaretaker{private: std::vector<Memento*> mementos; Originator* originator;public:Caretaker(Originator* orig):originator(orig){}~Caretaker(){for(auto m : mementos)delete m;}voidSave(){ mementos.push_back(originator->CreateMemento());}voidUndo(){if(mementos.empty())return; Memento* m = mementos.back(); originator->RestoreFromMemento(m); mementos.pop_back();delete m;}};// 使用示例intmain(){ Originator originator; Caretaker caretaker(&originator); originator.SetState("状态1"); caretaker.Save(); originator.SetState("状态2"); caretaker.Save(); originator.SetState("状态3");// 执行撤销 caretaker.Undo();// 恢复到状态2 caretaker.Undo();// 恢复到状态1return0;}

典型应用场景

备忘录模式在以下场景中特别有用:

  1. 撤销/重做功能:如文本编辑器中的撤销操作【1†source】【2†source】
  2. 事务回滚系统:数据库操作的事务管理【1†source】【5†source】
  3. 游戏存档系统:保存和恢复游戏进度【1†source】【2†source】
  4. 配置管理:保存和恢复系统配置状态

高级特性与优化

1. 增量备忘录

对于状态变化不大的场景,可以只保存变化部分【1†source】,这能显著减少内存消耗【1†source】。

2. 序列化支持

通过序列化技术,可以将备忘录状态持久化存储【2†source】,如保存到文件或数据库中【1†source】。

3. 线程安全考虑

在多线程环境中使用时【2†source】,需要添加适当的同步机制来保证状态的一致性【1†source】。

与其他模式的协作

备忘录模式常与其他设计模式配合使用:

  1. 命令模式:实现可撤销的操作【1†source】【2†source】
  2. 原型模式:通过克隆对象状态来优化备忘录创建【1†source】【2†source】
  3. 状态模式:保存不同状态配置【1†source】【2†source】

注意事项

  1. 内存消耗:对于大对象状态,需考虑内存使用情况【8†source】,可能影响系统性能【1†source】
  2. C++内存管理:特别注意指针和内存泄漏问题【1†source】【2†source】
  3. 替代方案:原型模式有时可作为更简单的替代方案【2†source】

总结

备忘录模式是面向对象设计原则的优秀实践【9†source】,通过状态封装实现了灵活的回滚机制【1†source】【2†source】。合理使用该模式可以显著提高系统的可维护性和扩展性【3†source】【7†source】。在C++中实现时,需要特别注意内存管理和线程安全等问题。掌握备忘录模式将帮助开发者构建更加强大、灵活的系统。

Read more

Linux命名管道(FIFO)通信:从原理到实操,一文搞懂跨进程通信

Linux命名管道(FIFO)通信:从原理到实操,一文搞懂跨进程通信

🔥个人主页:Cx330🌸 ❄️个人专栏:《C语言》《LeetCode刷题集》《数据结构-初阶》《C++知识分享》 《优选算法指南-必刷经典100题》《Linux操作系统》:从入门到入魔 《Git深度解析》:版本管理实战全解 🌟心向往之行必能至 🎥Cx330🌸的简介: 目录 前言: 一、先搞懂:命名管道(FIFO)是什么? 1. 命名管道的本质 2. 命名管道的核心特点 3. 命名管道与匿名管道的对比 二. 命名管道的创建方式 2.1 命令行创建(mkfifo 命令) 2.2 代码创建(mkfifo 函数) 2.3 命名管道的打开规则 三、实操实现:手搓命名管道通信 3.1 前置准备(

By Ne0inhk
Flutter 三方库 flutter_chat_ui 的鸿蒙化适配指南 - 掌握标准化的即时通讯 UI 构建技术、助力鸿蒙社交应用实现极致流畅且高度专业的对话交互体系

Flutter 三方库 flutter_chat_ui 的鸿蒙化适配指南 - 掌握标准化的即时通讯 UI 构建技术、助力鸿蒙社交应用实现极致流畅且高度专业的对话交互体系

欢迎加入开源鸿蒙跨平台社区:https://openharmonycrossplatform.ZEEKLOG.net Flutter 三方库 flutter_chat_ui 的鸿蒙化适配指南 - 掌握标准化的即时通讯 UI 构建技术、助力鸿蒙社交应用实现极致流畅且高度专业的对话交互体系 前言 在 OpenHarmony 鸿蒙应用全场景覆盖、特别是适配“即时通讯(IM)、在线客服系统、社区互动平台”等高频交互场景中,聊天界面的质感直接决定了用户的活跃度与留存率。一个成熟的聊天 UI 需要处理海量消息的平滑滚动、消息气泡的高低自适应、多种多媒体附件的预览以及极度复杂的键盘交互。如何从零构建这样一个工业级的界面?flutter_chat_ui 作为一个专注于“通过声明式声明快速产出专业级聊天 UI”的库,旨在为鸿蒙开发者提供一套标准的 IM 界面解决方案。本文将详述其在鸿蒙端的实战技法。 一、原原理分析 / 概念介绍 1.1 基础原理

By Ne0inhk
【OpenHarmony】鸿蒙Flutter混合开发实战指南

【OpenHarmony】鸿蒙Flutter混合开发实战指南

鸿蒙Flutter混合开发实战指南 概述 鸿蒙Flutter混合开发方案将Flutter的跨端UI能力与ArkTS的原生系统能力深度融合,实现高效的全场景应用开发。本文基于Flutter 3.24+、ArkTS 4.3,系统讲解混合开发的架构设计、双向通信实现、页面嵌入和原子化服务打包上架。 欢迎加入开源鸿蒙跨平台社区:https://openharmonycrossplatform.ZEEKLOG.net 技术架构设计 混合开发优势分析 技术框架核心优势适用场景ArkTS深度调用鸿蒙原生能力、性能接近原生系统级功能开发、高性能模块、原子化服务入口Flutter一次编码多端运行、UI渲染效率高、生态丰富跨设备通用UI、业务逻辑复用、快速迭代场景混合开发兼顾跨端效率与原生能力、低成本迁移全场景应用、跨平台+鸿蒙特色功能融合 两种核心开发模式 /// 混合开发模式枚举enumHybridMode{/// Flutter为主,ArkTS为辅/// 适用:存量Flutter项目适配鸿蒙 flutterPrimary,/// ArkTS为主,Flutter为辅/// 适

By Ne0inhk
【docker】新手必看!天下苦 Docker 安装久矣!Kali/Ubuntu 最新 Docker CE 安装,新手零门槛,拒绝复杂化!

【docker】新手必看!天下苦 Docker 安装久矣!Kali/Ubuntu 最新 Docker CE 安装,新手零门槛,拒绝复杂化!

一、安装前准备(必做) 1. 系统更新与依赖清理 先更新系统包,卸载旧版 Docker(避免冲突): # 更新系统包索引和已安装包 sudo apt update && sudo apt upgrade -y # 卸载旧版 Docker(如有) sudo apt remove -y docker docker.io containerd runc sudo apt autoremove -y 2. 安装必备依赖工具 sudo apt install -y ca-certificates curl gnupg lsb-release 二、安装 Docker CE(2026及以后

By Ne0inhk