0

0

C++如何实现备忘录模式 C++备忘录模式的设计

裘德小鎮的故事

裘德小鎮的故事

发布时间:2025-07-10 13:13:02

|

541人浏览过

|

来源于php中文网

原创

备忘录模式是一种保存和恢复对象状态的设计模式,其核心在于通过备忘录类存储对象状态,发起人类负责创建和恢复状态,管理者类用于管理多个备忘录。1. 使用模板实现通用备忘录类,避免类型限制;2. 采用智能指针(如 std::shared_ptr)管理内存,防止内存泄漏;3. 注意深拷贝对象状态,确保备忘录数据独立;4. 在多线程环境下使用互斥锁(std::mutex)保证线程安全;5. 性能优化可通过增量式保存、共享状态、延迟复制或移动语义等方式实现。该模式广泛应用于撤销/重做功能、事务回滚、游戏存档和配置版本管理等场景。

C++如何实现备忘录模式 C++备忘录模式的设计

备忘录模式,简单来说,就是把对象的状态保存下来,以后可以恢复到之前的状态。这在很多场景下都很有用,比如游戏存档、撤销操作等等。C++实现备忘录模式,核心在于定义一个备忘录类,用于存储对象的状态,以及一个发起人类,负责创建和恢复备忘录。

C++如何实现备忘录模式 C++备忘录模式的设计

C++备忘录模式的设计

C++如何实现备忘录模式 C++备忘录模式的设计

如何在C++中设计一个通用的备忘录类?

设计通用的备忘录类,要考虑到不同对象的状态可能差异很大。一种常见的做法是使用void*指针来存储状态,并配合一个类型标识符。但是,这种方式需要手动管理内存,容易出错。更安全的方式是使用模板,让备忘录类可以存储任意类型的数据。

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

C++如何实现备忘录模式 C++备忘录模式的设计
#include <iostream>
#include <memory>

template <typename T>
class Memento {
public:
    Memento(T state) : state_(std::make_shared<T>(state)) {}

    std::shared_ptr<T> getState() const {
        return state_;
    }

private:
    std::shared_ptr<T> state_;
};

template <typename T>
class Originator {
public:
    Originator(T state) : state_(state) {}

    void setState(T state) {
        state_ = state;
        std::cout << "State set to: " << state_ << std::endl;
    }

    Memento<T> saveStateToMemento() {
        std::cout << "Saving state to Memento: " << state_ << std::endl;
        return Memento<T>(state_);
    }

    void getStateFromMemento(Memento<T> memento) {
        state_ = *memento.getState();
        std::cout << "State after restoring from Memento: " << state_ << std::endl;
    }

private:
    T state_;
};

class Caretaker {
public:
    void add(Memento<int> state) {
        mementos_.push_back(state);
    }

    Memento<int> get(int index) const {
        return mementos_[index];
    }

private:
    std::vector<Memento<int>> mementos_;
};


int main() {
    Originator<int> originator(1);
    Caretaker caretaker;

    originator.setState(2);
    caretaker.add(originator.saveStateToMemento());

    originator.setState(3);
    caretaker.add(originator.saveStateToMemento());

    originator.setState(4);
    std::cout << "Current State: " << 4 << std::endl;

    originator.getStateFromMemento(caretaker.get(1));
    std::cout << "First saved State: " << originator.getStateFromMemento(caretaker.get(0)) << std::endl;

    return 0;
}

这段代码展示了如何使用模板来实现一个通用的备忘录模式。 Memento 类存储状态,Originator 类负责设置和恢复状态,Caretaker 类负责管理备忘录。 注意 std::shared_ptr 的使用,避免内存泄漏。

备忘录模式在实际项目中的应用场景有哪些?

备忘录模式在实际项目中有很多应用场景。

  • 撤销/重做功能: 这是最常见的应用。编辑器、绘图软件等,都需要支持撤销和重做操作。每次操作前,保存对象的状态到备忘录,撤销时,从备忘录中恢复。
  • 事务管理: 数据库事务需要保证原子性。如果事务失败,需要回滚到之前的状态。备忘录模式可以用来保存事务开始前的状态。
  • 游戏存档: 游戏玩家可以随时保存游戏进度。备忘录模式可以用来保存游戏状态。
  • 配置管理: 应用程序的配置可以有很多版本。备忘录模式可以用来保存不同版本的配置,方便切换。

如何优化C++备忘录模式的性能?

备忘录模式的性能瓶颈主要在于状态的复制。如果对象的状态很大,复制的开销会很大。可以考虑以下优化方法:

飞书多维表格
飞书多维表格

表格形态的AI工作流搭建工具,支持批量化的AI创作与分析任务,接入DeepSeek R1满血版

下载
  • 增量式备忘录: 只保存状态的变化部分,而不是完整状态。
  • 共享状态: 如果多个对象的状态相同,可以共享同一个备忘录。
  • 延迟复制: 在真正需要恢复状态时,才进行复制。
  • 使用移动语义: C++11引入了移动语义,可以避免不必要的复制。

例如,如果对象的状态是一个很大的数组,可以只保存数组中被修改的部分。

备忘录模式与其他设计模式的区别是什么?

备忘录模式容易与其他设计模式混淆,特别是与命令模式和状态模式。

  • 备忘录模式 vs. 命令模式: 命令模式将请求封装成对象,可以用于记录和撤销操作。备忘录模式则是保存对象的状态,用于恢复。命令模式关注的是操作,备忘录模式关注的是状态。
  • 备忘录模式 vs. 状态模式: 状态模式允许对象在内部状态改变时改变它的行为。备忘录模式则是保存对象的状态,用于外部恢复。状态模式关注的是对象内部的状态转换,备忘录模式关注的是对象外部的状态保存。

简单来说,命令模式是“做什么”,备忘录模式是“什么样”,状态模式是“我是谁”。

C++中实现备忘录模式有哪些常见的坑?

在C++中实现备忘录模式,容易遇到一些坑:

  • 内存管理: 如果使用void*指针存储状态,需要手动管理内存,容易出现内存泄漏或野指针。使用智能指针可以避免这个问题。
  • 深拷贝 vs. 浅拷贝: 备忘录需要保存对象的状态,必须进行深拷贝,而不是浅拷贝。否则,修改原始对象的状态,会影响备忘录中的状态。
  • 线程安全: 如果多个线程同时访问备忘录,需要考虑线程安全问题。可以使用互斥锁来保护备忘录的状态。

例如,如果备忘录类只是简单地复制了对象的指针,而不是复制对象本身,那么修改原始对象的状态,备忘录中的状态也会被修改。这就是浅拷贝的问题。

如何在多线程环境下使用备忘录模式?

在多线程环境下使用备忘录模式,需要保证线程安全。一种常见的做法是使用互斥锁来保护备忘录的状态。

#include <iostream>
#include <mutex>
#include <thread>

// ... (省略前面的代码)

template <typename T>
class Originator {
public:
    Originator(T state) : state_(state) {}

    void setState(T state) {
        std::lock_guard<std::mutex> lock(mutex_);
        state_ = state;
        std::cout << "State set to: " << state_ << " by thread " << std::this_thread::get_id() << std::endl;
    }

    Memento<T> saveStateToMemento() {
        std::lock_guard<std::mutex> lock(mutex_);
        std::cout << "Saving state to Memento: " << state_ << " by thread " << std::this_thread::get_id() << std::endl;
        return Memento<T>(state_);
    }

    void getStateFromMemento(Memento<T> memento) {
        std::lock_guard<std::mutex> lock(mutex_);
        state_ = *memento.getState();
        std::cout << "State after restoring from Memento: " << state_ << " by thread " << std::this_thread::get_id() << std::endl;
    }

private:
    T state_;
    std::mutex mutex_;
};

// ... (省略后面的代码)

int main() {
    Originator<int> originator(1);
    Caretaker caretaker;

    std::thread t1([&]() {
        originator.setState(2);
        caretaker.add(originator.saveStateToMemento());
    });

    std::thread t2([&]() {
        originator.setState(3);
        caretaker.add(originator.saveStateToMemento());
    });

    t1.join();
    t2.join();

    originator.setState(4);
    std::cout << "Current State: " << 4 << std::endl;

    originator.getStateFromMemento(caretaker.get(0));

    return 0;
}

这段代码使用 std::mutex 来保护 Originator 类的状态。每个访问状态的方法都需要获取锁,以保证线程安全。 注意,过度使用锁会影响性能,需要根据实际情况进行权衡。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
mysql标识符无效错误怎么解决
mysql标识符无效错误怎么解决

mysql标识符无效错误的解决办法:1、检查标识符是否被其他表或数据库使用;2、检查标识符是否包含特殊字符;3、使用引号包裹标识符;4、使用反引号包裹标识符;5、检查MySQL的配置文件等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

210

2023.12.04

Python标识符有哪些
Python标识符有哪些

Python标识符有变量标识符、函数标识符、类标识符、模块标识符、下划线开头的标识符、双下划线开头、双下划线结尾的标识符、整型标识符、浮点型标识符等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

322

2024.02.23

java标识符合集
java标识符合集

本专题整合了java标识符相关内容,想了解更多详细内容,请阅读下面的文章。

292

2025.06.11

c++标识符介绍
c++标识符介绍

本专题整合了c++标识符相关内容,阅读专题下面的文章了解更多详细内容。

178

2025.08.07

javascriptvoid(o)怎么解决
javascriptvoid(o)怎么解决

javascriptvoid(o)的解决办法:1、检查语法错误;2、确保正确的执行环境;3、检查其他代码的冲突;4、使用事件委托;5、使用其他绑定方式;6、检查外部资源等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

186

2023.11.23

java中void的含义
java中void的含义

本专题整合了Java中void的相关内容,阅读专题下面的文章了解更多详细内容。

134

2025.11.27

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

765

2023.08.10

Python 多线程与异步编程实战
Python 多线程与异步编程实战

本专题系统讲解 Python 多线程与异步编程的核心概念与实战技巧,包括 threading 模块基础、线程同步机制、GIL 原理、asyncio 异步任务管理、协程与事件循环、任务调度与异常处理。通过实战示例,帮助学习者掌握 如何构建高性能、多任务并发的 Python 应用。

377

2025.12.24

C# ASP.NET Core微服务架构与API网关实践
C# ASP.NET Core微服务架构与API网关实践

本专题围绕 C# 在现代后端架构中的微服务实践展开,系统讲解基于 ASP.NET Core 构建可扩展服务体系的核心方法。内容涵盖服务拆分策略、RESTful API 设计、服务间通信、API 网关统一入口管理以及服务治理机制。通过真实项目案例,帮助开发者掌握构建高可用微服务系统的关键技术,提高系统的可扩展性与维护效率。

76

2026.03.11

热门下载

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

精品课程

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

共94课时 | 11.2万人学习

C 教程
C 教程

共75课时 | 5.4万人学习

C++教程
C++教程

共115课时 | 21.6万人学习

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

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