【设计模式】如果再回到从前 ---- 备忘录模式

2024-04-05 01:32

本文主要是介绍【设计模式】如果再回到从前 ---- 备忘录模式,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一,概述

        定义:在不破坏封闭的前提下,捕获一个对象的内部状态,并在该对象之外保存这个状态。这样以后就可将该对象恢复到原先保存的状态。

         1.Originator(发起人):负责创建一个备忘录Memento,用以记录当前时刻自身的内部状态,并可使用备忘录恢复内部状态。Originator可以根据需要决定Memento存储自己的哪些内部状态。

  2.Memento(备忘录): 负责存储Originator对象的内部状态,并可以防止Originator以外的其他对象访问备忘录。备忘录有两个接口: Caretaker只能看到备忘录的窄接口,他只能将备忘录传递给其他对象。 Originator却可看到备忘录的宽接口,允许它访问返回到先前状态所需要的所有数据。
  3.Caretaker(管理者): 负责备忘录Memento不能对Memento的内容进行访问或者操作

二,基本代码格式

#include <iostream>
using namespace std;
class Memento//备忘录,存储 Originator内部状态 
{
private: 
string state;
public:
Memento(string state)
{
this->state = state;
}
string getState() 
{
return state; 
}
}; 
class Originator//发起人,创建备忘录 
{
private:
string state;
public: 
string getState() 
{
return state;
} 
void setState(string value)
{ 
this->state = value;
}
Memento* CreateMemento()//创建备忘录 
{
return (new Memento(state));
}
void SetMemento(Memento *memento)
{
this->state = memento->getState();
}
void Show()//显示备忘录状态 
{
cout<<"State="<<state<<endl;
}
}; 
class Caretaker//管理者 
{
public:
Memento *memento;
public:
Memento* getMemento()//返回备忘录状态 
{
return memento; 
} 
Memento* setMemento(Memento *mymemento)//设定备忘录状态 
{
this->memento=mymemento; 
} 
}; 
int main()
{
Originator *o = new Originator();//发起人 
o->setState("On");
o->Show();
Caretaker *c = new Caretaker();//管理者 
c->setMemento(o->CreateMemento());//保存状态 
o->setState("Off");//当游戏挂掉 
o->Show();
o->SetMemento(c->getMemento());//恢复之前保存的备忘录状态 
o->Show();
}


三,示例(存储游戏进度)

       玩游戏的时候,通常为了可以重复玩某一关关卡,在闯关之前先保存游戏状态。然后,假如没有闯关成功,再恢复到之前的状态重新闯关。


1)最原始的存储游戏进度方法

      方法:新建立一个游戏角色类,用以保存状态。

                  当需要恢复状态的时候,让状态存储类的数据恢复到游戏角色类中

     缺点:暴漏了实现细节

#include <iostream>
using namespace std; 
class GameRole
{
//生命力
private:
int vit;
public:
int getVitality() 
{
return vit; 
}
void   setVitality(int value) 
{ 
this->vit = value; 
}
//攻击力
private:
int atk;
public:
int getAttack() 
{
return atk; 
} 
void setAttack(int value) 
{
this->atk = value; 
}
//防御力
private:
int def;
public: 
int getDefense() 
{
return def;
} 
void setDefense(int value) 
{
this->def = value;
} 
//状态显示
void StateDisplay()
{
cout<<"角色当前状态:"<<endl;
cout<<"体力:"<<this->vit<<endl;
cout<<"攻击力:"<<this->atk<<endl;
cout<<"防御力:"<<this->def<<endl;
}
//获得初始状态
void GetInitState()
{
this->vit = 100;
this->atk = 100;
this->def = 100;
}
//战斗
void Fight()
{
this->vit = 0;
this->atk = 0;
this->def = 0;
}
}; 
int main()
{
//大战Boss前
GameRole *lixiaoyao = new GameRole();
lixiaoyao->GetInitState();
lixiaoyao->StateDisplay();
//保存进度
GameRole *backup = new GameRole();
backup->setVitality(lixiaoyao->getVitality());
backup->setAttack(lixiaoyao->getAttack());
backup->setDefense(lixiaoyao->getDefense());
//大战Boss时,损耗严重
lixiaoyao->Fight();
lixiaoyao->StateDisplay();
//恢复之前状态
lixiaoyao->setVitality(backup->getVitality());
lixiaoyao->setAttack(backup->getAttack());
lixiaoyao->setDefense(backup->getDefense());
lixiaoyao->StateDisplay();
}

2)备忘录模式

       优点:客户端隐藏了实现细节。

#include <iostream>
using namespace std; 
//角色状态存储箱
class RoleStateMemento
{
private: int vit;
int atk;
int def;
public:
RoleStateMemento(int vit, int atk, int def)
{
this->vit = vit;
this->atk = atk;
this->def = def;
}
//生命力
int getVitality() 
{
return vit; 
}
void   setVitality(int value) 
{ 
this->vit = value; 
}
//攻击力       
int getAttack() 
{
return atk; 
} 
void setAttack(int value) 
{
this->atk = value; 
}
//防御力
int getDefense() 
{
return def;
} 
void setDefense(int value) 
{
this->def = value;
} 
}; 
class GameRole
{
//生命力
private:
int vit;
public:
int getVitality() 
{
return vit; 
}
void   setVitality(int value) 
{ 
this->vit = value; 
}
//攻击力
private:
int atk;
public:
int getAttack() 
{
return atk; 
} 
void setAttack(int value) 
{
this->atk = value; 
}
//防御力
private:
int def;
public: 
int getDefense() 
{
return def;
} 
void setDefense(int value) 
{
this->def = value;
} 
//状态显示
void StateDisplay()
{
cout<<"角色当前状态:"<<endl;
cout<<"体力:"<<this->vit<<endl;
cout<<"攻击力:"<<this->atk<<endl;
cout<<"防御力:"<<this->def<<endl;
}
//保存角色状态
RoleStateMemento *SaveState()
{
return (new RoleStateMemento(vit, atk, def));
}
//恢复角色状态
void RecoveryState(RoleStateMemento *memento)
{
this->vit = memento->getVitality();
this->atk = memento->getAttack();
this->def = memento->getDefense();
}
//获得初始状态
void GetInitState()
{
this->vit = 100;
this->atk = 100;
this->def = 100;
}
//战斗
void Fight()
{
this->vit = 0;
this->atk = 0;
this->def = 0;
}
}; 
//角色状态管理者
class RoleStateCaretaker
{
private:
RoleStateMemento *memento;
public:
RoleStateMemento* getMemento() 
{
return memento; 
}
void setMemento(RoleStateMemento *value) 
{ 
this->memento = value; 
}
}; 
int main()
{
//大战Boss前
GameRole *lixiaoyao = new GameRole();
lixiaoyao->GetInitState();
lixiaoyao->StateDisplay();
//保存进度
RoleStateCaretaker *stateAdmin = new RoleStateCaretaker();
stateAdmin->setMemento(lixiaoyao->SaveState());
//大战Boss时,损耗严重
lixiaoyao->Fight();
lixiaoyao->StateDisplay();
//恢复之前状态
lixiaoyao->RecoveryState(stateAdmin->getMemento());
lixiaoyao->StateDisplay();
}















这篇关于【设计模式】如果再回到从前 ---- 备忘录模式的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/877332

相关文章

Redis Cluster模式配置

《RedisCluster模式配置》:本文主要介绍RedisCluster模式配置,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录分片 一、分片的本质与核心价值二、分片实现方案对比 ‌三、分片算法详解1. ‌范围分片(顺序分片)‌2. ‌哈希分片3. ‌虚

RabbitMQ工作模式中的RPC通信模式详解

《RabbitMQ工作模式中的RPC通信模式详解》在RabbitMQ中,RPC模式通过消息队列实现远程调用功能,这篇文章给大家介绍RabbitMQ工作模式之RPC通信模式,感兴趣的朋友一起看看吧... 目录RPC通信模式概述工作流程代码案例引入依赖常量类编写客户端代码编写服务端代码RPC通信模式概述在R

Mac备忘录怎么导出/备份和云同步? Mac备忘录使用技巧

《Mac备忘录怎么导出/备份和云同步?Mac备忘录使用技巧》备忘录作为iOS里简单而又不可或缺的一个系统应用,上手容易,可以满足我们日常生活中各种记录的需求,今天我们就来看看Mac备忘录的导出、... 「备忘录」是 MAC 上的一款常用应用,它可以帮助我们捕捉灵感、记录待办事项或保存重要信息。为了便于在不同

SQL Server身份验证模式步骤和示例代码

《SQLServer身份验证模式步骤和示例代码》SQLServer是一个广泛使用的关系数据库管理系统,通常使用两种身份验证模式:Windows身份验证和SQLServer身份验证,本文将详细介绍身份... 目录身份验证方式的概念更改身份验证方式的步骤方法一:使用SQL Server Management S

Redis高可用-主从复制、哨兵模式与集群模式详解

《Redis高可用-主从复制、哨兵模式与集群模式详解》:本文主要介绍Redis高可用-主从复制、哨兵模式与集群模式的使用,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝... 目录Redis高可用-主从复制、哨兵模式与集群模式概要一、主从复制(Master-Slave Repli

一文带你搞懂Redis Stream的6种消息处理模式

《一文带你搞懂RedisStream的6种消息处理模式》Redis5.0版本引入的Stream数据类型,为Redis生态带来了强大而灵活的消息队列功能,本文将为大家详细介绍RedisStream的6... 目录1. 简单消费模式(Simple Consumption)基本概念核心命令实现示例使用场景优缺点2

Nginx location匹配模式与规则详解

《Nginxlocation匹配模式与规则详解》:本文主要介绍Nginxlocation匹配模式与规则,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、环境二、匹配模式1. 精准模式2. 前缀模式(不继续匹配正则)3. 前缀模式(继续匹配正则)4. 正则模式(大

Linux系统配置NAT网络模式的详细步骤(附图文)

《Linux系统配置NAT网络模式的详细步骤(附图文)》本文详细指导如何在VMware环境下配置NAT网络模式,包括设置主机和虚拟机的IP地址、网关,以及针对Linux和Windows系统的具体步骤,... 目录一、配置NAT网络模式二、设置虚拟机交换机网关2.1 打开虚拟机2.2 管理员授权2.3 设置子

SpringBoot如何通过Map实现策略模式

《SpringBoot如何通过Map实现策略模式》策略模式是一种行为设计模式,它允许在运行时选择算法的行为,在Spring框架中,我们可以利用@Resource注解和Map集合来优雅地实现策略模式,这... 目录前言底层机制解析Spring的集合类型自动装配@Resource注解的行为实现原理使用直接使用M

C#原型模式之如何通过克隆对象来优化创建过程

《C#原型模式之如何通过克隆对象来优化创建过程》原型模式是一种创建型设计模式,通过克隆现有对象来创建新对象,避免重复的创建成本和复杂的初始化过程,它适用于对象创建过程复杂、需要大量相似对象或避免重复初... 目录什么是原型模式?原型模式的工作原理C#中如何实现原型模式?1. 定义原型接口2. 实现原型接口3