通过命令模式封装操作并结合备忘录模式保存状态,实现可撤销的命令。具体步骤为:1. 定义命令接口,包含execute和undo方法;2. 命令执行前保存接收者状态到备忘录;3. 执行时记录命令到历史栈;4. 撤销时调用栈顶命令的undo方法并恢复状态;5. 清理资源防止内存泄漏。该方式适用于文本编辑器等需多级撤销的应用,结构清晰且扩展性强。
在C++中实现可撤销的命令,可以通过结合命令模式(Command Pattern)和备忘录模式(Memento Pattern)来完成。命令模式负责封装操作,使其可以像对象一样被传递、存储和执行;而备忘录模式则用于保存对象的状态,以便后续恢复,从而支持撤销操作。
命令模式将每个操作封装成一个独立的对象,包含执行(execute)和撤销(undo)方法。这样,我们可以把命令放入历史栈中,在需要时调用 undo 恢复之前的状态。
例如,假设我们有一个文本编辑器,支持插入文本的操作:
class TextEditor {
public:
void insert(const std::string& text) {
content += text;
}
void deleteLast(int length) {
if (length <= content.size()) {
content.erase(content.size() - length);
}
}
std::string getState() const { return content; }
void setState(const std::string& state) { content = state; }
private:
std::string content;
};
接下来定义命令接口:
class Command {
public:
virtual ~Command() = default;
virtual void execute() = 0;
virtual void undo() = 0;
};
备忘录模式通过一个“备忘录”对象保存原发器(如 TextEditor)的内部状态,并提供有限访问权限,以维护封装性。
为 TextEditor 添加创建和恢复状态的能力:
class Memento {
public:
Memento(const std::string& state) : state_(state) {}
std::string getState() const { return state_; }
private:
std::string state_;
};
// 在 TextEditor 中添加:
Memento createMemento() const {
return new Memento(content);
}
void restoreFromMemento(Memento m) {
content = m->getState();
delete m; // 可选:也可由外部管理生命周期
}
现在定义具体的可撤销命令。在执行前保存当前状态,undo 时恢复该状态。
class InsertCommand : public Command {
public:
InsertCommand(TextEditor* editor, const std::string& text)
: editor_(editor), text_(text) {}
void execute() override {
memento_ = editor_->createMemento(); // 执行前保存状态
editor_->insert(text_);
}
void undo()
override {
if (memento_) {
editor_->restoreFromMemento(memento_);
}
}private:
TextEditor editor;
std::string text;
Memento memento_ = nullptr;
};
使用命令的历史管理器来支持多级撤销:
class CommandHistory {
public:
void push(Command* command) {
command->execute();
history.push_back(command);
}
void undo() {
if (!history.empty()) {
Command* command = history.back();
command->undo();
history.pop_back();
delete command;
}
}
~CommandHistory() {
for (auto cmd : history) {
delete cmd;
}
}private:
std::vector history;
};
将这些组件组合起来:
int main() {
TextEditor editor;
CommandHistory history;
history.push(new InsertCommand(&editor, "Hello"));
std::cout << editor.getState() << "\n"; // 输出: Hello
history.push(new InsertCommand(&editor, " World"));
std::cout << editor.getState() << "\n"; // 输出: Hello World
history.undo();
std::cout << editor.getState() << "\n"; // 输出: Hello
history.undo();
std::cout << editor.getState() << "\n"; // 输出: (空)
return 0;
}
每次执行命令前保存状态,undo 时恢复,实现了可靠的撤销功能。
基本上就这些。关键在于命令对象持有执行前的备忘录,在 undo 时利用它还原状态。这种方式结构清晰,扩展性强,适合需要多级撤销/重做的应用,比如编辑器、图形软件或游戏指令系统。
# 栈
# ai
# c++
# String
# 封装
# 接口
# private
# 对象
# history
# 编辑器
# 适用于
# 我们可以
# 可以通过
# 使其
# 可选
# 管理器
# 它还
# 来完成
# 关键在于
相关文章:
,购物网站怎么盈利呢?
如何配置IIS站点权限与局域网访问?
建站之星如何助力网站排名飙升?揭秘高效技巧
如何续费美橙建站之星域名及服务?
网站制作企业,网站的banner和导航栏是指什么?
如何基于云服务器快速搭建个人网站?
建站主机默认首页配置指南:核心功能与访问路径优化
上海制作企业网站有哪些,上海有哪些网站可以让企业免费发布招聘信息?
建站主机选哪家性价比最高?
如何选择网络建站服务器?高效建站必看指南
如何获取PHP WAP自助建站系统源码?
制作网站的软件下载免费,今日头条开宝箱老是需要下载怎么回事?
网站制作与设计教程,如何制作一个企业网站,建设网站的基本步骤有哪些?
安徽网站建设与外贸建站服务专业定制方案
网站制作费用多少钱,一个网站的运营,需要哪些费用?
微网站制作教程,我微信里的网站怎么才能复制到浏览器里?
微信小程序 input输入框控件详解及实例(多种示例)
枣阳网站制作,阳新火车站打的到仙岛湖多少钱?
建站之星后台管理如何实现高效配置?
学校为何禁止电信移动建设网站?
北京的网站制作公司有哪些,哪个视频网站最好?
如何高效生成建站之星成品网站源码?
弹幕视频网站制作教程下载,弹幕视频网站是什么意思?
C++ static_cast和dynamic_cast区别_C++静态转换与动态类型安全转换
c++23 std::expected怎么用 c++优雅处理函数错误返回【详解】
香港服务器租用每月最低只需15元?
阿里云网站制作公司,阿里云快速搭建网站好用吗?
如何快速生成凡客建站的专业级图册?
学校免费自助建站系统:智能生成+拖拽设计+多端适配
如何制作网站标识牌,动态网站如何制作(教程)?
ppt制作免费网站有哪些,ppt模板免费下载网站?
如何用5美元大硬盘VPS安全高效搭建个人网站?
娃派WAP自助建站:免费模板+移动优化,快速打造专业网站
装修招标网站设计制作流程,装修招标流程?
上海网站制作网页,上海本地的生活网站有哪些?最好包括生活的各个方面的?
手机钓鱼网站怎么制作视频,怎样拦截钓鱼网站。怎么办?
招商网站制作流程,网站招商广告语?
青浦网站制作公司有哪些,苹果官网发货地是哪里?
建站之星在线客服如何快速接入解答?
深圳网站制作费用多少钱,读秀,深圳文献港这样的网站很多只提供网上试读,但有些人只要提供试读的文章就能全篇下载,这个是怎么弄的?
专业网站制作企业网站,如何制作一个企业网站,建设网站的基本步骤有哪些?
简单实现Android验证码
怎么制作网站设计模板图片,有电商商品详情页面的免费模板素材网站推荐吗?
详解jQuery中基本的动画方法
如何在IIS中新建站点并配置端口与物理路径?
北京网站制作费用多少,建立一个公司网站的费用.有哪些部分,分别要多少钱?
如何彻底卸载建站之星软件?
网站制作多少钱一个,建一个论坛网站大约需要多少钱?
建站之星安装后如何配置SEO及设计样式?
Python路径拼接规范_跨平台处理说明【指导】
*请认真填写需求信息,我们会在24小时内与您取得联系。