- 使用 Spring Initializr 创建 Spring Boot 应用程序
- 在Spring Boot中配置Cassandra
- 在 Spring Boot 上配置 Tomcat 连接池
- 将Camel消息路由到嵌入WildFly的Artemis上
备忘录模式(Memento Design Pattern),也叫快照(Snapshot)模式。指在不违背封装原则前提下,捕获一个对象的内部状态,并在该对象之外保存这个状态,以便之后恢复对象为先前的状态。
备忘录模式在日常中很常见,比如Word中的回退,MySQL中的undo log
日志,Git版本管理等等,我们都可以从当前状态退回之前保存的状态。比如Git中的checkout
命令就可以从main
版本切换到之前的bugFix
版本:
备忘录是一种对象行为型模式,它提供了一种可以恢复状态的机制,并实现了内部状态的封装。下面就来看看备忘录模式的结构及其对应的实现:
备忘录的核心是备忘录类(Memento)和管理备忘录的管理者类(Caretaker)的设计,其结构如下图所示:
Originator
:组织者类,记录当前业务的状态信息,提供备忘录创建和恢复的功能Memento
:备忘录类,存储组织者类的内部状态,在需要时候提供这些内部状态给组织者类Caretaker
:管理者类,对备忘录进行管理,提供存储于获取备忘录的功能,无法对备忘录对象进行修改和访问在利用备忘录模式时,首先应该设计一个组织者类(Originator),它是一个具体的业务类,存储当前状态。它包含备忘录对象的创建方法createMemeto()
和备忘录对象恢复方法restoreMemeto()
。
Originator
类的具体代码如下:
public class Originator {
private String state;
public String getState() {
return state;
}
public void setState(String state) {
this.state = state;
}
//创建一个备忘录对象
public Memento createMemento() {
return new Memento(this);
}
//根据备忘录对象,恢复之前组织者的状态
public void restoreMemento(Memento m) {
state = m.getState();
}
}
对于备忘录类(Memento)而言,它存储组织者类(Originator)的状态,其具体代码如下:
public class Memento {
private String state;
public Memento(Originator o) {
this.state = state;
}
public String getState() {
return state;
}
public void setState(String state) {
this.state = state;
}
}
在这里需要考虑备忘录的封装性,除了Originator
类外,其他类不能调用备忘录的内部的相关方法。因为外界类的调用可能会引起备忘录内的状态发生变化,这样备忘录的设置就没有了意义。在实际操作中,可以将Memento
和Originator
类定义在同一个包中来实现封装;也可以将Memento
类作为Originator
的内部类。
下面再了看看管理者类(Caretaker)的具体代码:
public class Caretaker {
private Memento memento;
public Memento getMemento() {
return memento;
}
public void setMemento(Memento memento) {
this.memento = memento;
}
}
它的作用仅仅是存储备忘录对象,而且其内部中也不应该有直接调用Memento
中的状态改变方法。只有当用户需要对Originator
类进行恢复时,再将存储在其中的备忘录对象取出。
下面是对整个流程的测试:
public class Client {
public static void main(String[] args) {
Originator originator = new Originator();
Caretaker caretaker = new Caretaker();
//在originator和caretaker中保存memento对象
originator.setState("1");
System.out.println("当前的状态是:" + originator.getState());
caretaker.setMemento(originator.createMemento());
originator.setState("2");
System.out.println("当前的状态是:" + originator.getState());
//从Caretaker取出Memento对象
originator.restoreMemento(caretaker.getMemento());
System.out.println("执行状态恢复,当前的状态是:" + originator.getState());
}
}
测试结果为:
当前的状态是:1
当前的状态是:2
执行状态恢复,当前的状态是:1
正如开头提到的,备忘录模式可以用在诸如Word文字编辑器,PhotoShop等软件的状态保存,还有数据库的备份等等场景。下面引用一个文本编辑的代码实现,来自于《设计模式》
/**
* @description: 输入text的当前状态
* @author: wjw
* @date: 2022/4/8
*/
public class InputText {
private StringBuilder text = new StringBuilder();
public StringBuilder getText() {
return text;
}
public void setText(StringBuilder text) {
this.text = text;
}
//创建SnapMemento对象
public SnapMemento createSnapMemento() {
return new SnapMemento(this);
}
//恢复SnapMemento对象
public void restoreSnapMemento(SnapMemento sm) {
text = sm.getText();
}
}
/**
* @description: 快照备忘录
* @author: wjw
* @date: 2022/4/8
*/
public class SnapMemento {
private StringBuilder text;
public SnapMemento(InputText it) {
text = it.getText();
}
public StringBuilder getText() {
return text;
}
public void setText(StringBuilder text) {
this.text = text;
}
}
/**
* @description: 负责SnapMemento对象的获取和存储
* @author: wjw
* @date: 2022/4/8
*/
public class SnapMementoHolder {
private Stack<SnapMemento> snapMementos = new Stack<>();
//获取snapMemento对象
public SnapMemento popSnapMemento() {
return snapMementos.pop();
}
//存储snapMemento对象
public void pushSnapMemento(SnapMemento sm) {
snapMementos.push(sm);
}
}
/**
* @description: 客户端
* @author: wjw
* @date: 2022/4/8
*/
public class test_memento {
public static void main(String[] args) {
InputText inputText = new InputText();
StringBuilder first_stringBuilder = new StringBuilder("First StringBuilder");
inputText.setText(first_stringBuilder);
SnapMementoHolder snapMementoHolder = new SnapMementoHolder();
snapMementoHolder.pushSnapMemento(inputText.createSnapMemento());
System.out.println("当前的状态是:" + inputText.getText().toString());
StringBuilder second_stringBuilder = new StringBuilder("Second StringBuilder");
inputText.setText(second_stringBuilder);
System.out.println("修改过后的状态是:" + inputText.getText().toString());
inputText.restoreSnapMemento(snapMementoHolder.popSnapMemento());
System.out.println("利用备忘录恢复的状态:" + inputText.getText().toString());
}
}
测试结果:
当前的状态是:First StringBuilder
修改过后的状态是:Second StringBuilder
利用备忘录恢复的状态:First StringBuilder
在本案例中模拟系统在发布上线的过程中记录线上配置文件用于紧急回滚(案例来源于《重学Java设计模式》):
其中配置文件中包含版本、时间、MD5、内容信息和操作人。如果一旦遇到紧急问题,系统可以通过回滚操作将配置文件回退到上一个版本中。那么备忘录存储的信息就是配置文件的内容,根据备忘录模式设计该结构:
ConfigMemento
:备忘录类,是对原有配置类的扩展ConfigOriginator
:记录者类,相当于之前的管理者(Caretaker),获取和返回备忘录对象Admin
:管理员类,操作修改备忘信息,相当于之前的组织者(Originator)ConfigFile
配置信息类public class ConfigFile {
private String versionNo;
private String content;
private Date dateTime;
private String operator;
//getset,constructor
}
ConfigMemento
备忘录类public class ConfigMemento {
private ConfigFile configFile;
public ConfigMemento(ConfigFile configFile) {
this.configFile = configFile;
}
public ConfigFile getConfigFile() {
return configFile;
}
public void setConfigFile(ConfigFile configFile) {
this.configFile = configFile;
}
}
ConfigOriginator
配置文件组织者类public class ConfigOriginator {
private ConfigFile configFile;
public ConfigFile getConfigFile() {
return configFile;
}
public void setConfigFile(ConfigFile configFile) {
this.configFile = configFile;
}
public ConfigMemento saveMemento() {
return new ConfigMemento(configFile);
}
public void getMemento(ConfigMemento memento) {
this.configFile = memento.getConfigFile();
}
}
Admin
配置文件管理者类public class Admin {
//版本信息
private int cursorIdx = 0;
private List<ConfigMemento> mementoList = new ArrayList<>();
private Map<String, ConfigMemento> mementoMap = new ConcurrentHashMap<String, ConfigMemento>();
//新增版本信息
public void append(ConfigMemento memento) {
mementoList.add(memento);
mementoMap.put(memento.getConfigFile().getVersionNo(), memento);
cursorIdx++;
}
//回滚历史配置
public ConfigMemento undo() {
if (--cursorIdx <= 0) {
return mementoList.get(0);
}
return mementoList.get(cursorIdx);
}
//前进历史配置
public ConfigMemento redo() {
if(++cursorIdx > mementoList.size()) {
return mementoList.get(mementoList.size() - 1);
}
return mementoList.get(cursorIdx);
}
public ConfigMemento get(String versionNo) {
return mementoMap.get(versionNo);
}
}
public class ApiTest {
private Logger logger = LoggerFactory.getLogger(ApiTest.class);
@Test
public void test_memento() {
Admin admin = new Admin();
ConfigOriginator configOriginator = new ConfigOriginator();
configOriginator.setConfigFile(new ConfigFile("1000001", "配置内容1", new Date(), "ethan"));
admin.append(configOriginator.saveMemento());
configOriginator.setConfigFile(new ConfigFile("1000002", "配置内容2", new Date(), "ethan"));
admin.append(configOriginator.saveMemento());
configOriginator.setConfigFile(new ConfigFile("1000003", "配置内容3", new Date(), "ethan"));
admin.append(configOriginator.saveMemento());
configOriginator.setConfigFile(new ConfigFile("1000004", "配置内容4", new Date(), "ethan"));
admin.append(configOriginator.saveMemento());
//(第一次回滚)
configOriginator.getMemento(admin.undo());
logger.info("回滚undo: {}", JSON.toJSONString(configOriginator.getConfigFile()));
//(第二次回滚)
configOriginator.getMemento(admin.undo());
logger.info("回滚undo: {}", JSON.toJSONString(configOriginator.getConfigFile()));
// (前进)
configOriginator.getMemento(admin.redo());
logger.info("前进redo:{}", JSON.toJSONString(configOriginator.getConfigFile()));
// (获取)
configOriginator.getMemento(admin.get("1000002"));
logger.info("获取get:{}", JSON.toJSONString(configOriginator.getConfigFile()));
}
}
测试结果:
22:44:39.773 [main] INFO ApiTest - 回滚undo: {"content":"配置内容4","dateTime":1649429079642,"operator":"ethan","versionNo":"1000004"}
22:44:39.777 [main] INFO ApiTest - 回滚undo: {"content":"配置内容3","dateTime":1649429079642,"operator":"ethan","versionNo":"1000003"}
22:44:39.777 [main] INFO ApiTest - 前进redo:{"content":"配置内容4","dateTime":1649429079642,"operator":"ethan","versionNo":"1000004"}
22:44:39.777 [main] INFO ApiTest - 获取get:{"content":"配置内容2","dateTime":1649429079642,"operator":"ethan","versionNo":"1000002"}
《Java设计模式》
《设计模式》
《重学Java设计模式》
关闭。这个问题需要更多focused .它目前不接受答案。 想改善这个问题吗?更新问题,使其仅关注一个问题 editing this post . 4年前关闭。 Improve this questi
.NET 框架:4.5.1 我在 Blend for visual studio 2015 中遇到一个奇怪的错误,我找不到它的来源。 如果我在 VS 中打开我的 WPF 解决方案,它会加载并运行良好。
我经常遇到这样的问题,与 Hierarchical RESTful URL design 非常相似 假设该服务仅提供用户上传文档。 POST, GET /accounts PUT, DELETE /a
在 Rails 应用程序中,我使用 devise 来管理我的用户,而我用来销毁 session 的链接不再有效。它正在工作,现在我添加了事件管理员,但没有。 我的链接是 :delete, :clas
我已经坚持了超过 24 小时,试图按照此处发布的其他解决方案进行操作,但我无法使其正常工作。我是 Rails 新手,需要帮助! 我想让我的/users/edit 页面正常工作,以便我可以简单地更改用户
Devise 在以下情况下不会使用户超时: 用户登录,关闭选项卡,然后在超时 + X 分钟内重新访问该 URL。用户仍处于登录状态。 如果选项卡已打开并且稍后刷新/单击,则超时可以正常工作。这意味着
我想使用这样的 slider 我希望该 slider 根据提供给它的值进行相应调整。到目前为止,我只能应用具有渐变效果的背景,但无法获得这种效果。请通过提供样式代码来帮助我。
您应该为每种方法创建一个请求/响应对象,还是应该为每个服务创建一个? 如果我在所有方法中使用它,我的服务请求对象中将只有 5 个不同的东西,因为我对几乎所有方法使用相同的输入。 响应对象将只有一个字典
我正在尝试在 REST 中对实体的附件进行建模。假设一个缺陷实体可以附加多个附件。每个附件都有描述和一些其他属性(上次修改时间、文件大小...)。附件本身是任何格式的文件(jpeg、doc ...)
我有以下表格: Blogs { BlogName } BlogPosts { BlogName, PostTitle } 博客文章同时建模一个实体和一个关系,根据 6nf(根据第三个宣言)这是无效的。
如果 A 类与 B、C 和 D 类中的每一个都有唯一的交互,那么交互的代码应该在 A 中还是在 B、C 和 D 中? 我正在编写一个小游戏,其中许多对象可以与其他对象进行独特的交互。例如,EMP点击
关于如何记住我与 Omniauth 一起工作似乎有些困惑。 根据这个wiki ,您需要在 OmniauthCallbacksController 中包含以下内容: remember_me(user)
设计问题: 使用 非线程安全 组件(集合,API,...)在/带有 多线程成分 ... 例子 : 组件 1 :多线程套接字服务器谁向消息处理程序发送消息... 组件 2 :非线程安全 消息处理程序 谁
我们目前正在设计一个 RESTful 应用程序。我们决定使用 XML 作为我们的基本表示。 我有以下关于在 XML 中设计/建模应用程序数据的问题。 在 XML 中进行数据建模的方法有哪些?从头开始然
我正在设计一个新的 XSD 来从业务合作伙伴那里获取积分信息。对于每笔交易,合作伙伴必须提供至少一种积分类型的积分值。我有以下几点:
设计支持多个版本的 API 的最佳方法是什么。我如何确保即使我的数据架构发生更改(微小更改),我的 api 的使用者也不会受到影响?任何引用架构、指南都非常有用。 最佳答案 Mark Nottingh
关闭。这个问题是opinion-based 。目前不接受答案。 想要改进这个问题吗?更新问题,以便 editing this post 可以用事实和引文来回答它。 . 已关闭 4 年前。 Improv
我想用 php 创建一个网站,其工作方式与 https://www.bitcoins.lc/ 相同。确实,就每个页面上具有相同布局但内容会随着您更改链接/页面而改变而言,我如何在 php 中使用lay
我有一个关于编写 Swing UI 的问题。如果我想制作一个带有某些选项的软件,例如在第一个框架上,我有三个按钮(新建、选项、退出)。 现在,如果用户单击新按钮,我想将框架中的整个内容更改为其他内容。
我正在尝试找出并学习将应用程序拥有的一堆Docker容器移至Kubernetes的模式和最佳实践。诸如Pod设计,服务,部署之类的东西。例如,我可以创建一个其中包含单个Web和应用程序容器的Pod,但
我是一名优秀的程序员,十分优秀!