备忘录模式是一种保存和恢复对象状态的设计模式,其核心在于通过备忘录类存储对象状态,发起人类负责创建和恢复状态,管理者类用于管理多个备忘录。1. 使用模板实现通用备忘录类,避免类型限制;2. 采用智能指针(如 std::shared_ptr)管理内存,防止内存泄漏;3. 注意深拷贝对象状态,确保备忘录数据独立;4. 在多线程环境下使用互斥锁(std::mutex)保证线程安全;5. 性能优化可通过增量式保存、共享状态、延迟复制或移动语义等方式实现。该模式广泛应用于撤销/重做功能、事务回滚、游戏存档和配置版本管理等场景。
备忘录模式,简单来说,就是把对象的状态保存下来,以后可以恢复到之前的状态。这在很多场景下都很有用,比如游戏存档、撤销操作等等。C++实现备忘录模式,核心在于定义一个备忘录类,用于存储对象的状态,以及一个发起人类,负责创建和恢复备忘录。
C++备忘录模式的设计
设计通用的备忘录类,要考虑到不同对象的状态可能差异很大。一种常见的做法是使用void*指针来存储状态,并配合一个类型标识符。但是,这种方式需要手动管理内存,容易出错。更安全的方式是使用模板,让备忘录类可以存储任意类型的数据。
立即学习“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++中实现备忘录模式,容易遇到一些坑:
例如,如果备忘录类只是简单地复制了对象的指针,而不是复制对象本身,那么修改原始对象的状态,备忘录中的状态也会被修改。这就是浅拷贝的问题。
在多线程环境下使用备忘录模式,需要保证线程安全。一种常见的做法是使用互斥锁来保护备忘录的状态。
#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中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号