全网整合营销服务商

电脑端+手机端+微信端=数据同步管理

免费咨询热线:400-708-3566

C++如何实现一个备忘录模式_C++设计模式之捕获对象状态并支持回滚

备忘录模式通过发起者创建、管理者保存、备忘录存储状态实现对象状态的保存与恢复,适用于撤销、回滚等场景,如文本编辑器;关键在于私有化备忘录构造函数并用友元保证封装性,使用栈管理多级撤销,注意内存开销与深拷贝问题。

在C++中实现备忘录模式,核心是捕获一个对象的内部状态,并在不破坏封装性的前提下将其保存,以便后续恢复。这种设计常用于需要支持撤销、回滚或历史记录功能的场景,比如文本编辑器、游戏存档、配置管理等。

备忘录模式的角色组成

备忘录模式包含三个基本角色:

  • 发起者(Originator):拥有内部状态,能够创建备忘录来保存当前状态,也能通过备忘录恢复状态。
  • 备忘录(Memento):负责存储发起者的内部状态。通常将构造函数设为私有,仅允许发起者访问其内容。
  • 管理者(Caretaker):负责保存和管理备忘录,但不能修改或查看其内容。

实现步骤与代码示例

以下是一个简单的C++实现,模拟文本编辑器的撤销功能:

#include 
#include 
#include 

class Memento;

// 发起者:文本编辑器
class TextEditor {
private:
    std::string content;

public:
    void write(const std::string& text) {
        content += text;
    }

    std::string getContent() const {
        return content;
    }

    // 创建备忘录
    Memento createMemento() const;

    // 恢复状态
    void restoreFromMemento(const Memento& m);
};

// 备忘录类
class Memento {
    friend class TextEditor; // 允许TextEditor访问私有成员

private:
    std::string state;

    // 私有构造函数,仅发起者可创建
    Memento(const std::string& state) : state(state) {}

public:
    std::string getState() const {
        return state;
    }
};

// 成员函数实现在Memento定义之后
Memento TextEditor::createMemento() const {
    return Memento(content);
}

void TextEditor::restoreFromMemento(const Memento& m) {
    content = m.state;
}

// 管理者:负责保存和恢复备忘录
class History {
private:
    std::stack snapshots;

public:
    void push(const Memento& m) {
        snapshots.push(m);
    }

    Memento pop() {
        if (snapshots.empty()) {
            throw std::runtime_error("No snapshots available");
        }
        Memento m = snapshots.top();
        snapshots.pop();
        return m;
    }

    bool empty() const {
        return snapshots.empty();
    }
};

使用示例:

int main() {
    TextEditor editor;
    History history;

    editor.write("Hello ");
    history.push(editor.createMemento()); // 保存状态

    editor.write("World!");
    std::cout << "Current: " << editor.getContent() << "\n";

    editor.restoreFromMemento(history.pop()); // 撤销
    std::cout << "After undo: " << editor.getContent() << "\n";

    return 0;
}

关键设计要点

实现时需注意以下几点以保证封装性和安全性:

  • Memento 的构造函数设为私有,防止外部直接构造非法状态。
  • 使用 friend 关键字让发起者能访问备忘录的私有数据,而管理者只能存储和传递,无法查看或修改。
  • 若状态较大,可考虑使用指针或共享所有权(如 std::shared_ptr)避免频繁拷贝。
  • 支持多级撤销时,可用栈结构保存多个备忘录;若需内存优化,可引入快照压缩或差量存储。

适用场景与注意事项

备忘录模式适合用于需要精确回滚到历史状态的场合。但它会带来一定的内存开销,尤其是状态大或保存频繁时。应根据实际需求控制保存频率,或设置最大保存数量。

如果对象状态无法被安全复制(如包含资源句柄),需实现深拷贝逻辑。对于复杂对象,也可结合序列化机制保存状态。

基本上就这些。只要把握好封装边界,C++中的备忘录模式可以很清晰地实现状态保存与恢复。


#   # ai  # c++  # ios  # stream  # 封装性  # red  # 封装  # 构造函数  # 指针  # 对象  # 编辑器  # 设为  # 用友  # 是一个  # 尤其是  # 多个  # 句柄  # 也能  # 适用于  # 并在 


相关文章: 建站主机是什么?如何选择适合的建站主机?  网站制作多少钱一个,建一个论坛网站大约需要多少钱?  设计网站制作公司有哪些,制作网页教程?  网站制作员失业,怎样查看自己网站的注册者?  Swift开发中switch语句值绑定模式  官网网站制作腾讯审核要多久,联想路由器newifi官网  Python如何创建带属性的XML节点  我的世界制作壁纸网站下载,手机怎么换我的世界壁纸?  网站制作哪家好,cc、.co、.cm哪个域名更适合做网站?  ppt在线制作免费网站推荐,有什么下载免费的ppt模板网站?  如何在阿里云虚拟机上搭建网站?步骤解析与避坑指南  高性能网站服务器部署指南:稳定运行与安全配置优化方案  如何在云主机快速搭建网站站点?  专业的网站制作设计是什么,如何制作一个企业网站,建设网站的基本步骤有哪些?  已有域名和空间,如何快速搭建网站?  宿州网站制作公司兴策,安徽省低保查询网站?  Android滚轮选择时间控件使用详解  建站之星客服服务时间及联系方式如何?  如何在Golang中引入测试模块_Golang测试包导入与使用实践  如何快速搭建虚拟主机网站?新手必看指南  如何快速辨别茅台真假?关键步骤解析  昆明高端网站制作公司,昆明公租房申请网上登录入口?  如何用手机制作网站和网页,手机移动端的网站能制作成中英双语的吗?  如何彻底卸载建站之星软件?  宝塔Windows建站如何避免显示默认IIS页面?  c# await 一个已经完成的Task会发生什么  ,制作一个手机app网站要多少钱?  如何通过cPanel快速搭建网站?  如何通过网站建站时间优化SEO与用户体验?  巅云智能建站系统:可视化拖拽+多端适配+免费模板一键生成  网站按钮制作软件,如何实现网页中按钮的自动点击?  制作网站的基本流程,设计网站的软件是什么?  c# 在高并发场景下,委托和接口调用的性能对比  广东企业建站网站优化与SEO营销核心策略指南  如何基于云服务器快速搭建个人网站?  建站之星如何防范黑客攻击与数据泄露?  企业微网站怎么做,公司网站和公众号有什么区别?  建站之星导航配置指南:自助建站与SEO优化全解析  如何用西部建站助手快速创建专业网站?  青浦网站制作公司有哪些,苹果官网发货地是哪里?  SAX解析器是什么,它与DOM在处理大型XML文件时有何不同?  如何在Windows服务器上快速搭建网站?  宝塔建站无法访问?如何排查配置与端口问题?  建站主机选虚拟主机还是云服务器更好?  网站制作难吗安全吗,做一个网站需要多久时间?  JS中使用new Date(str)创建时间对象不兼容firefox和ie的解决方法(两种)  企业网站制作费用多少,企业网站空间一般需要多大,费用是多少?  网站制作新手教程,新手建设一个网站需要注意些什么?  c++23 std::expected怎么用 c++优雅处理函数错误返回【详解】  装修招标网站设计制作流程,装修招标流程? 

您的项目需求

*请认真填写需求信息,我们会在24小时内与您取得联系。