首页 > 后端开发 > C++ > 正文

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

裘德小鎮的故事
发布: 2025-07-10 13:13:02
原创
478人浏览过

备忘录模式是一种保存和恢复对象状态的设计模式,其核心在于通过备忘录类存储对象状态,发起人类负责创建和恢复状态,管理者类用于管理多个备忘录。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++备忘录模式的性能?

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

  • 增量式备忘录: 只保存状态的变化部分,而不是完整状态。
  • 共享状态: 如果多个对象的状态相同,可以共享同一个备忘录。
  • 延迟复制: 在真正需要恢复状态时,才进行复制。
  • 使用移动语义: 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 类的状态。每个访问状态的方法都需要获取锁,以保证线程安全。 注意,过度使用锁会影响性能,需要根据实际情况进行权衡。

以上就是C++如何实现备忘录模式 C++备忘录模式的设计的详细内容,更多请关注php中文网其它相关文章!

最佳 Windows 性能的顶级免费优化软件
最佳 Windows 性能的顶级免费优化软件

每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。

下载
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
最新问题
开源免费商场系统广告
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板
关于我们 免责申明 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送
PHP中文网APP
随时随地碎片化学习
PHP中文网抖音号
发现有趣的

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