【再谈设计模式】备忘录模式~对象状态的守护者

news/2025/3/3 8:18:00/


一、引言

        在软件开发过程中,我们常常会遇到需要保存对象状态以便在之后恢复的情况。例如,在文本编辑器中,我们可能想要撤销之前的操作;在游戏中,玩家可能希望恢复到之前的某个游戏状态。备忘录模式(Memento Pattern)就为这种需求提供了一种有效的解决方案。


二、定义与描述

        备忘录模式属于行为型设计模式。它的主要目的是在不破坏对象封装性的前提下,捕获并外部化一个对象的内部状态,以便之后可以将该对象恢复到这个状态。这个模式涉及到三个主要角色:

        原发器(Originator):创建一个备忘录,用于记录当前时刻它的内部状态。原发器还可以使用备忘录来恢复其内部状态。

        备忘录(Memento):存储原发器对象的内部状态。备忘录应该防止原发器以外的其他对象访问其内部状态。

        负责人(Caretaker):负责保存备忘录,但不能对备忘录的内容进行操作或检查。

 


三、抽象背景

        在许多应用场景中,对象的状态会随着时间发生变化。然而,我们可能需要在某个特定的时间点保存对象的状态,以便后续可以撤销操作或者回到之前的某个状态。如果直接将对象的状态暴露给外部进行保存和恢复,会破坏对象的封装性。备忘录模式通过引入一个专门的备忘录对象来解决这个问题,使得对象的状态可以在不破坏封装性的情况下被保存和恢复。


四、适用场景与现实问题解决

(一)适用场景

        文本编辑器的撤销/重做功能:当用户输入文字、删除文字或者进行格式调整时,每一个操作都可以看作是对象(文本内容)状态的改变。通过备忘录模式,可以轻松地保存每个操作前的文本状态,从而实现撤销和重做功能。

        游戏中的存档和读档功能:游戏中的各种元素(角色属性、游戏场景等)构成了对象的状态。玩家在游戏过程中可能希望在某些关键节点保存游戏状态,之后可以随时读取之前保存的状态继续游戏。

(二)现实问题解决

        以文本编辑器为例,假设我们有一个TextDocument类作为原发器。在用户进行编辑操作(如插入字符、删除字符等)之前,我们可以创建一个备忘录来保存当前文档的状态。如果用户执行了撤销操作,我们可以从备忘录中恢复文档的之前状态。这样就可以在不破坏TextDocument类内部结构的情况下,实现撤销功能。


五、备忘录模式的现实生活的例子

        考虑拍照的过程。相机可以看作是原发器,照片就是备忘录。当我们按下快门(相当于创建备忘录)时,相机的当前状态(镜头聚焦、光圈大小、曝光设置等)被记录在照片上。我们可以将照片存储起来(相当于负责人保存备忘录),之后如果想要回顾某个瞬间(相当于恢复到之前的状态),我们可以查看对应的照片。

 


六、初衷与问题解决

        初衷是在保持对象封装性的同时,实现对象状态的保存和恢复。通过将对象状态的保存和恢复逻辑封装在备忘录模式中,避免了外部对象直接操作对象的内部状态,从而提高了代码的可维护性和安全性。


七、代码示例

(一)Java

// 备忘录类
class Memento {private String state;public Memento(String state) {this.state = state;}public String getState() {return state;}
}// 原发器类
class Originator {private String state;public void setState(String state) {this.state = state;}public Memento saveToMemento() {return new Memento(state);}public void restoreFromMemento(Memento memento) {state = memento.getState();}
}// 负责人类
class Caretaker {private Memento memento;public void saveMemento(Memento memento) {this.memento = memento;}public Memento getMemento() {return memento;}
}public class Main {public static void main(String[] args) {Originator originator = new Originator();originator.setState("State 1");Caretaker caretaker = new Caretaker();caretaker.saveMemento(originator.saveToMemento());originator.setState("State 2");originator.restoreFromMemento(caretaker.getMemento());System.out.println(originator.state);}
}

类图:

流程图:

 

时序图:

 


(二)C++

#include <iostream>
#include <string>// 备忘录类
class Memento {
private:std::string state;
public:Memento(std::string state) : state(state) {}std::string getState() const { return state; }
};// 原发器类
class Originator {
private:std::string state;
public:void setState(std::string state) { this->state = state; }Memento saveToMemento() { return Memento(state); }void restoreFromMemento(const Memento& memento) { state = memento.getState(); }
};// 负责人类
class Caretaker {
private:Memento memento;
public:void saveMemento(const Memento& memento) { this->memento = memento; }Memento getMemento() const { return memento; }
};int main() {Originator originator;originator.setState("State 1");Caretaker caretaker;caretaker.saveMemento(originator.saveToMemento());originator.setState("State 2");originator.restoreFromMemento(caretaker.getMemento());std::cout << originator.state << std::endl;return 0;
}

(三)Python

# 备忘录类
class Memento:def __init__(self, state):self.state = statedef get_state(self):return self.state# 原发器类
class Originator:def __init__(self):self.state = Nonedef set_state(self, state):self.state = statedef save_to_memento(self):return Memento(self.state)def restore_from_memento(self, memento):self.state = memento.get_state()# 负责人类
class Caretaker:def __init__(self):self.memento = Nonedef save_memento(self, memento):self.memento = mementodef get_memento(self):return self.mementoif __name__ == "__main__":originator = Originator()originator.set_state("State 1")caretaker = Caretaker()caretaker.save_memento(originator.save_to_memento())originator.set_state("State 2")originator.restore_from_memento(caretaker.get_memento())print(originator.state)

(四)Go

package mainimport "fmt"// 备忘录结构体
type Memento struct {state string
}func NewMemento(state string) *Memento {return &Memento{state: state}
}func (m *Memento) getState() string {return m.state
}// 原发器结构体
type Originator struct {state string
}func (o *Originator) setState(state string) {o.state = state
}func (o *Originator) saveToMemento() *Memento {return NewMemento(o.state)
}func (o *Originator) restoreFromMemento(m *Memento) {o.state = m.getState()
}// 负责人结构体
type Caretaker struct {memento *Memento
}func (c *Caretaker) saveMemento(m *Memento) {c.memento = m
}func (c *Caretaker) getMemento() *Memento {return c.memento
}func main() {originator := Originator{}originator.setState("State 1")caretaker := Caretaker{}caretaker.saveMemento(originator.saveToMemento())originator.setState("State 2")originator.restoreFromMemento(caretaker.getMemento())fmt.Println(originator.state)
}

八、备忘录模式的优缺点

(一)优点

        保持封装性:原发器的内部状态对外部是隐藏的,只有原发器自身能够访问备忘录中的状态,从而保护了对象的封装性。

        简化撤销/恢复操作:通过备忘录模式,撤销和恢复操作可以很容易地实现,不需要在原发器中编写复杂的状态管理代码。

        提供状态历史记录:可以方便地保存多个备忘录,从而形成对象状态的历史记录,这对于需要查看对象状态变化历史的应用场景非常有用。

(二)缺点

        资源消耗:如果需要保存大量的备忘录,可能会消耗较多的内存资源,尤其是当备忘录对象包含大量数据时。

        管理复杂度:随着备忘录数量的增加,备忘录的管理(如存储、查找、删除等)可能会变得复杂。

特性描述
优点
保持封装性原发器的内部状态对外部是隐藏的,只有原发器自身能够访问备忘录中的状态,从而保护了对象的封装性。
简化撤销/恢复操作通过备忘录模式,撤销和恢复操作可以很容易地实现,不需要在原发器中编写复杂的状态管理代码。
提供状态历史记录可以方便地保存多个备忘录,从而形成对象状态的历史记录,这对于需要查看对象状态变化历史的应用场景非常有用。
缺点
资源消耗如果需要保存大量的备忘录,可能会消耗较多的内存资源,尤其是当备忘录对象包含大量数据时。
管理复杂度随着备忘录数量的增加,备忘录的管理(如存储、查找、删除等)可能会变得复杂。

九、备忘录模式的升级版

        一种常见的升级版是增加一个历史列表(History List)管理类,它可以管理多个备忘录对象,并且提供更方便的操作,如按照时间顺序查看备忘录、限制备忘录的数量以避免资源过度消耗等。例如,在文本编辑器的撤销/重做功能中,这个历史列表可以存储多个编辑操作的备忘录,并且可以方便地根据用户的操作(如多次撤销或重做)找到对应的备忘录进行状态恢复。

 


http://www.ppmy.cn/news/1576242.html

相关文章

NIM平台开发基于提示工程的大语言模型(LLM)应用

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 1 课程介绍1.1 Goals1.2 content 2 提示词简介2.1 NVIDIA NIMs 用于提示工程2.2 OpenAI API 交互2.3 与 LangChain 交互实现聊天2.4 流式处理和批处理2.5 迭代式提示…

《白帽子讲 Web 安全:点击劫持》

目录 摘要&#xff1a; 一、点击劫持概述 二、点击劫持的实现示例&#xff1a;诱导用户收藏指定淘宝商品 案例 构建恶意页面&#xff1a; 设置绝对定位和z - index&#xff1a; 控制透明度&#xff1a; 三、其他相关攻击技术 3.1图片覆盖攻击与 XSIO 3.2拖拽劫持与数据…

大白话实战docker

为什么有docker 假设有这么一个场景,作为新时代的全栈工程师,某一天你接手一个项目,搭建好项目以后,还需要一台服务器,在它上面安装mysql、redis等项目中常用的中间件,于是你网上搜索Linux如何安装mysql、redis,照着网上的资料进行复制粘贴,结果发现老是运行不成功,好…

23种设计模式一览【设计模式】

文章目录 前言一、创建型模式&#xff08;Creational Patterns&#xff09;二、结构型模式&#xff08;Structural Patterns&#xff09;三、行为型模式&#xff08;Behavioral Patterns&#xff09; 前言 设计模式是软件工程中用来解决特定问题的一组解决方案。它们是经过验证…

vector 面试点总结

ps&#xff1a;部分内容使用“AI”查询 一、入门 1、什么是vector 动态数组容器&#xff0c;支持自动扩容、随机访问和连续内存存储。 2、怎么创建-初始化vector std::vector<int> v; // 创建空vectorstd::vector<int> v {1, 2, 3}; // 直接初始化std::vec…

在Ubuntu中,某个文件的右下角有一把锁的标志是什么意思?

在Ubuntu中&#xff0c;某个文件的右下角有一把锁的标志是什么意思&#xff1f; 在 Ubuntu&#xff08;或其他基于 GNOME 文件管理器的 Linux 发行版&#xff09;中&#xff0c;文件或文件夹的右下角出现一把“锁”标志&#xff0c;通常表示 你当前的用户没有该文件/文件夹的写…

Go语言学习笔记(五)

文章目录 十八、go操作MySQL、RedisMySQLRedis 十九、泛型泛型函数泛型类型泛型约束泛型特化泛型接口 二十、workspaces核心概念示例 二十一、模糊测试 十八、go操作MySQL、Redis MySQL package mainimport ("database/sql""errors""fmt"_ &qu…

ChatGPT 提示词框架

作为一个资深安卓开发工程师&#xff0c;我们在日常开发中经常会用到 ChatGPT 来提升开发效率&#xff0c;比如代码优化、bug 排查、生成单元测试等。 但要想真正发挥 ChatGPT 的潜力&#xff0c;我们需要掌握一些提示词&#xff08;Prompt&#xff09;的编写技巧&#xff0c;并…