Go设计模式--备忘录模式,带暂存的业务功能可以参考它来实现

开发 前端
备忘录模式是一种行为型设计模式。这种模式允许我们保存对象在某些关键节点时的必要信息,以便于在适当的时候可以将之恢复到之前的状态。通常它可以用来帮助设计撤销/恢复操作。

大家好,这里是每周都在陪你一起进步的网管~!今天继续学习设计模式—备忘录模式

备忘录模式(Memento Pattern)又叫作快照模式(Snapshot Pattern), 或令牌模式(Token Pattern), 指在不破坏封装的前提下, 捕获一个对象的内部状态, 并在对象之外保存这个状态。 这样以后就可将该对象恢复到原先保存的状态, 属于行为型设计模式。

备忘录模式主要适用于以下应用场景。

  1. 需要保存历史快照的场景。
  2. 希望在对象之外保存状态,且除了自己,其他类对象无法访问状态保存的具体内容。

备忘录模式是一种行为型设计模式。这种模式允许我们保存对象在某些关键节点时的必要信息,以便于在适当的时候可以将之恢复到之前的状态。通常它可以用来帮助设计撤销/恢复操作。

备忘录模式的类结构

下面是备忘录模式的类图,

图片来自https://refactoringguru.cn/design-patterns/memento,我后面实现的时候不会完全按照这个结构去实现,这里先把结构里的各个角色给大家说清楚。

图片

备忘录模式中主要有这三个角色的类

  • Originator(发起者):Originator是当前的基础对象,它会将自己的状态保存进备忘录,此角色可以类比博客系统中的文章对象
  • 发起者中要有保存方法和从备忘录中恢复状态的方法,保存方法会返回当时状态组成的备忘录对象
  • Memento(备忘录) : 存储着Originator的状态的对象,类比理解即为文章对象的不同版本。
  • Caretaker(管理人):Caretaker是保存着多条备忘录的对象,并维护着备忘录的索引,在需要的时候会返回相应的备忘录 -- 类比理解为博客系统中的编辑器对象
  • 管理者的保存和恢复操作,会代理其持有的发起者对象的保存和恢复操作,在这些代理方法中会增加对备忘录对象列表、当前备忘录版本的维护。

上面这个类图结构是实现备忘录模式的最简单方式,真实使用的时候,Caretaker,Originator、memento 这些角色可以继续抽象出对应的接口和实现。这里就不搞那么复杂了,要举的例子比较简单,这么一拆显得这个模式用起来特别费事儿。

其实其他设计模式也是一样,学习的时候大家知道了它的结构后,在实现应用的环节不必完完全全按照结构一板一眼地全部进行实现,有的应用场景并不复杂,能合并的角色可以按需进行合并。

应用举例

场景

某线上博客平台, 需为用户提供在线编辑文章功能,文章主要包括标题 - title 和内容 - content等信息。为最大程度防止异常情况导致编辑内容的丢失, 需要提供版本暂存和Undo, Redo功能。

"版本暂存"问题可以应用备忘录模式, 将编辑器的状态完整保存起来(主要就是编辑内容),Undo和Redo的本质, 是在历史版本中前后移动把当时保存的内容加载到文章对象上。

方案设计

  • IEditor: 定义编辑器接口
  • EditorMemento: 定义编辑器的备忘录, 也就是编辑器的内部状态数据模型, 同时也对应一个历史版本
  • Editor: 编辑器类, 实现IEditor接口

图片

这个例子里我们把原发器和管理人两个角色集中在Editor类型上一起实现,例子比较简单,就没有单独实现一个Article类型作为原发器角色,如果你想完全按照备忘录模式的结构实现,把Title、Content这写属性和Save方法抽离到单独的Article类型上,再让Editor嵌套组合Article即可。

下面我们根据UML类图实现一下这个带Undo、Redo功能的编辑器。

代码实现

首先在IEditor 接口里定义编辑器对象要实现的行为

// 编辑器接口定义
type IEditor interface {
Title(title string)
Content(content string)
Save()
Undo() error
Redo() error
Show()
}

接下来定义编辑器的备忘录, 也就是编辑器的内部状态数据模型, 同时也对应一个历史版本。

"本文使用的完整可运行源码
去公众号「网管叨bi叨」发送【设计模式】即可领取"
type Memento struct {
title string
content string
createTime int64
}

func newMemento(title string, content string) *Memento {
return &Memento{
title, content, time.Now().Unix(),
}
}

然后是最复杂的Editor实现,它会实现上面IEditor接口中定义的所有行为,其中的Undo、Redo方法即回退、前进方法在实现的时候就是依赖的它内部记录的一组Memento对象,通过指向不同的Memento对象来实现回退和前进功能。

"本文使用的完整可运行源码
去公众号「网管叨bi叨」发送【设计模式】即可领取"
// 编辑器类, 实现IEditor接口
type Editor struct {
title string
content string
versions []*Memento
index int
}

func NewEditor() IEditor {
return &Editor{
"", "", make([]*Memento, 0), 0,
}
}

func (editor *Editor) Title(title string) {
editor.title = title
}

func (editor *Editor) Content(content string) {
editor.content = content
}

func (editor *Editor) Save() {
it := newMemento(editor.title, editor.content)
editor.versions = append(editor.versions, it)
editor.index = len(editor.versions) - 1
}

func (editor *Editor) Undo() error {
return editor.load(editor.index - 1)
}

func (editor *Editor) load(i int) error {
size := len(editor.versions)
if size <= 0 {
return errors.New("no history versions")
}

if i < 0 || i >= size {
return errors.New("no more history versions")
}

it := editor.versions[i]
editor.title = it.title
editor.content = it.content
editor.index = i
return nil
}

func (editor *Editor) Redo() error {
return editor.load(editor.index + 1)
}

func (editor *Editor) Show() {
fmt.Printf("MockEditor.Show, title=%s, cnotallow=%s\n", editor.title, editor.content)
}

最后我们来测试一下Editor的版本记录功能

"本文使用的完整可运行源码
去公众号「网管叨bi叨」发送【设计模式】即可领取"
func main() {
editor := NewEditor()

// test save()
editor.Title("唐诗")
editor.Content("白日依山尽")
editor.Save()

editor.Title("唐诗 登鹳雀楼")
editor.Content("白日依山尽, 黄河入海流. ")
editor.Save()

editor.Title("唐诗 登鹳雀楼 王之涣")
editor.Content("白日依山尽, 黄河入海流。欲穷千里目, 更上一层楼。")
editor.Save()

// test show()
fmt.Println("-------------Editor 当前内容-----------")
editor.Show()

fmt.Println("-------------Editor 回退内容-----------")
for {
e := editor.Undo()
if e != nil {
break
} else {
editor.Show()
}
}

fmt.Println("-------------Editor 前进内容-----------")
for {
e := editor.Redo()
if e != nil {
break
} else {
editor.Show()
}
}
}

运行程序后会有类似下面的显示

图片


责任编辑:武晓燕 来源: 网管叨bi叨
相关推荐

2020-11-02 10:41:33

备忘录模式

2023-10-07 00:14:53

2023-10-31 09:07:16

备忘录模式保存

2018-08-02 17:39:42

iPhone备忘录iOS

2022-04-07 08:00:00

Javascript开发

2018-12-24 21:40:12

2023-10-10 15:26:30

内存泄露OOM

2014-04-17 10:30:41

Linux 命令黑白备忘录

2017-03-21 11:02:59

基础深度学习备忘录

2011-04-11 10:03:32

钱伯斯思科

2011-08-16 18:38:23

Core Animat动画

2023-05-04 08:47:31

命令模式抽象接口

2018-06-20 13:14:16

MySQL数据优化查询备忘录

2013-08-29 10:50:48

移动网站性能优化移动web

2023-04-10 09:20:13

设计模式访客模式

2021-11-07 23:41:39

Windows 11Windows微软

2011-12-07 09:19:49

JavaJ2MEBicaVM

2021-03-08 00:12:44

Grid 备忘录 函数

2023-09-27 12:28:08

Kubernetes命令

2014-06-30 14:36:40

瞻博普天
点赞
收藏

51CTO技术栈公众号