实例讲解JAVA设计模式之备忘录模式

 更新时间:2020年06月24日 15:18:06   作者:认真对待  
这篇文章主要介绍了JAVA设计模式之备忘录模式的的相关资料,文中示例代码非常详细,供大家参考和学习,感兴趣的朋友可以了解下

在讲述这个模式之前,我们先看一个案例:游戏回档

游戏的某个场景,一游戏角色有生命力、攻击力、防御力等数据,在打Boss前和后会不一样,我们允许玩家如果感觉与Boss决斗的效果不理想,可以让游戏恢复到决斗前。下面是代码:

游戏角色类,用来存储角色的生命力、攻击力、防御力的数据。

public class GameRole {
 private int vit;//生命力
 private int atk;//攻击力
 private int def;//防御力
 
 //状态显示
 public void stateDisplay() {
  System.out.println("当前角色状态:");
  System.out.println("体力:"+this.vit);
  System.out.println("攻击力"+this.atk);
  System.out.println("防御力"+this.def);
 }
 //获取初始状态
 public void getInitState() {
  //数据通常来自本地磁盘或远程数据库
  this.vit = 100;
  this.atk = 100;
  this.def = 100;
 }
 //战斗
 public void fight() {
  //在与Boss大战后游戏数据损耗为0
  this.vit = 0;
  this.atk = 0;
  this.def = 0;
 }
 //省略getter、setter方法
 
}
//测试方法
public class Test {
 public static void main(String[] args) {
  //大战Boss前
  GameRole lixiaoyao = new GameRole();
  lixiaoyao.getInitState();//Boss大战前,获得角色初始状态
  lixiaoyao.stateDisplay();
  
  //保存进度,通过游戏角色的新实例来保存进度
  GameRole backup = new GameRole();
  backup.setVit(lixiaoyao.getVit());
  backup.setAtk(lixiaoyao.getAtk());
  backup.setDef(lixiaoyao.getDef());
  
  //大战Boss时,损耗严重,所有数据全部损耗为0
  lixiaoyao.fight();
  lixiaoyao.stateDisplay();
  
  //恢复之前状态,重新来玩
  lixiaoyao.setVit(backup.getVit());
  lixiaoyao.setAtk(backup.getAtk());
  lixiaoyao.setDef(backup.getDef());
  lixiaoyao.stateDisplay();
  
 }
}

上面的代码实现了效果,但是不理想的是:main方法里暴露了太多“细节”,使得main方法需要知道“生命力、攻击力、防御力”这样的细节。以后需要增加“魔法值”或修改现有的“生命力”为“经验值”,这部分就要修改了。同样的道理也存在于恢复时的代码。显然,我们希望的是把这些“游戏角色”的存取状态细节封装起来,而且最好是封装在外部的类中。以体现职责分离。

下面介绍备忘录模式:https://www.jb51.net/article/189469.htm

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

用备忘录模式优化案例

public class GameRole {
 private int vit;//生命力
 private int atk;//攻击力
 private int def;//防御力
 
 //状态显示
 public void stateDisplay() {
  System.out.println("当前角色状态:");
  System.out.println("体力:"+this.vit);
  System.out.println("攻击力"+this.atk);
  System.out.println("防御力"+this.def);
 }
 //获取初始状态
 public void getInitState() {
  //数据通常来自本地磁盘或远程数据库
  this.vit = 100;
  this.atk = 100;
  this.def = 100;
 }
 //战斗
 public void fight() {
  //在与Boss大战后游戏数据损耗为0
  this.vit = 0;
  this.atk = 0;
  this.def = 0;
 }
 //新增“保存角色状态”方法,将游戏角色的三个状态值通过实例化“角色状态存储箱”返回
 public RoleStateMemento saveState() {
  return new RoleStateMemento(vit, atk, def);
 }
 
 //新增“恢复角色状态”方法,可将外部的“角色状态存储箱”中的状态值恢复给游戏角色
 public void recoveryState(RoleStateMemento memento) {
  this.vit = memento.getAtk();
  this.atk = memento.getAtk();
  this.def = memento.getDef();
 }
 
 //省略getter、setter方法
 
}
//角色状态存储箱类
public class RoleStateMemento {
 private int vit;//生命力
 private int atk;//攻击力
 private int def;//防御力
 //将生命力、攻击力、防御力存入状态存储箱对象中
 public RoleStateMemento(int vit, int atk, int def) {
  super();
  this.vit = vit;
  this.atk = atk;
  this.def = def;
 }
 
 //省略getter、setter方法
 
}
//角色状态管理者类
public class RoleStateCaretaker {
 private RoleStateMemento memento;

 public RoleStateMemento getMemento() {
  return memento;
 }

 public void setMemento(RoleStateMemento memento) {
  this.memento = memento;
 }
 
}
//测试方法
public class Test {
 public static void main(String[] args) {
  //大战Boss前
  GameRole lixiaoyao = new GameRole();
  lixiaoyao.getInitState();//Boss大战前,获得角色初始状态
  lixiaoyao.stateDisplay();
  
  //保存进度,由于封装在Memento中,因此我们并不知道保存了哪些具体的数据
  RoleStateCaretaker stateAdmin = new RoleStateCaretaker();
  stateAdmin.setMemento(lixiaoyao.saveState());
  
  //大战Boss时,损耗严重
  lixiaoyao.fight();
  lixiaoyao.stateDisplay();
  
  //恢复之前的状态
  lixiaoyao.recoveryState(stateAdmin.getMemento());
  lixiaoyao.stateDisplay();
  
 }
}

输出结果同上。

肯定有人会问:对于“角色状态”的保存,直接调用RoleStateMemento进行set和get不就行了,为什么还需要一个RoleStateCaretaker类呢?

这是为了符合迪米特法则进行的优化!

备忘录模式也是有缺点的,角色状态需要完整存储到备忘录对象中,如果状态数据很大很多,那么在资源消耗上,备忘录对象会非常耗内存。所以也不是用的越多越好。

以上就是实例讲解JAVA设计模式之备忘录模式的详细内容,更多关于JAVA 备忘录模式的资料请关注脚本之家其它相关文章!

相关文章

  • Mybatis一对多关联关系映射实现过程解析

    Mybatis一对多关联关系映射实现过程解析

    这篇文章主要介绍了Mybatis一对多关联关系映射实现过程解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-02-02
  • SpringBoot+aop实现主从数据库的读写分离操作

    SpringBoot+aop实现主从数据库的读写分离操作

    读写分离的作用是为了缓解写库,也就是主库的压力,但一定要基于数据一致性的原则,就是保证主从库之间的数据一定要一致,这篇文章给大家介绍SpringBoot+aop实现主从数据库的读写分离操作,感兴趣的朋友跟随小编一起看看吧
    2024-03-03
  • Mybatis-plus 查询条件为空不生效问题及解决

    Mybatis-plus 查询条件为空不生效问题及解决

    这篇文章主要介绍了Mybatis-plus 查询条件为空不生效问题及解决,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-01-01
  • ssm mybatis如何配置多个mapper目录

    ssm mybatis如何配置多个mapper目录

    这篇文章主要介绍了ssm mybatis如何配置多个mapper目录,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教。
    2022-01-01
  • Java List中数据的去重

    Java List中数据的去重

    今天小编就为大家分享一篇关于Java List中数据的去重,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧
    2019-01-01
  • Java中Redis存储String类型会有乱码的问题及解决方案

    Java中Redis存储String类型会有乱码的问题及解决方案

    在java中使用Redis存储String类型的数据时,会出现乱码,我写了一条存储key为name,值为虎哥的字符串,然后获取一下这个key为name的值,打印得到的值,下面通过实例代码介绍Java中Redis存储String类型会有乱码的问题及解决方案,一起看看吧
    2024-04-04
  • JavaStream将List转为Map示例

    JavaStream将List转为Map示例

    这篇文章主要为大家介绍了JavaStream将List转为Map示例,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-09-09
  • Java中的Unsafe工具类使用详解

    Java中的Unsafe工具类使用详解

    这篇文章主要介绍了Java中的Unsafe工具类使用详解,Unsafe是jdk提供的一个直接访问操作系统资源的工具类(底层c++实现),它可以直接分配内存,内存复制,copy,提供cpu级别的CAS乐观锁等操作,需要的朋友可以参考下
    2023-12-12
  • 学习不同 Java.net 语言中类似的函数结构

    学习不同 Java.net 语言中类似的函数结构

    这篇文章主要介绍了学习不同 Java.net 语言中类似的函数结构,函数式编程语言包含多个系列的常见函数。但开发人员有时很难在语言之间进行切换,因为熟悉的函数具有不熟悉的名称。函数式语言倾向于基于函数范例来命名这些常见函数。,需要的朋友可以参考下
    2019-06-06
  • java中的定时器和多线程

    java中的定时器和多线程

    这篇文章主要介绍了java中的定时器和多线程用法,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2024-01-01

最新评论