全网整合营销服务商

电脑端+手机端+微信端=数据同步管理

免费咨询热线:400-708-3566

Java设计模式之备忘录模式_动力节点Java学院

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

类型:行为类

类图:

        我们在编程的时候,经常需要保存对象的中间状态,当需要的时候,可以恢复到这个状态。比如,我们使用Eclipse进行编程时,假如编写失误(例如不小心误删除了几行代码),我们希望返回删除前的状态,便可以使用Ctrl+Z来进行返回。这时我们便可以使用备忘录模式来实现。

备忘录模式的结构

  • 发起人:记录当前时刻的内部状态,负责定义哪些属于备份范围的状态,负责创建和恢复备忘录数据。
  • 备忘录:负责存储发起人对象的内部状态,在需要的时候提供发起人需要的内部状态。
  • 管理角色:对备忘录进行管理,保存和提供备忘录。

通用代码实现

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.state); 
  } 
  public void restoreMemento(Memento memento){ 
    this.setState(memento.getState()); 
  } 
} 
class Memento { 
  private String state = ""; 
  public Memento(String state){ 
    this.state = state; 
  } 
  public String getState() { 
    return state; 
  } 
  public void setState(String state) { 
    this.state = state; 
  } 
} 
class Caretaker { 
  private Memento memento; 
  public Memento getMemento(){ 
    return memento; 
  } 
  public void setMemento(Memento memento){ 
    this.memento = memento; 
  } 
} 
public class Client { 
  public static void main(String[] args){ 
    Originator originator = new Originator(); 
    originator.setState("状态1"); 
    System.out.println("初始状态:"+originator.getState()); 
    Caretaker caretaker = new Caretaker(); 
    caretaker.setMemento(originator.createMemento()); 
    originator.setState("状态2"); 
    System.out.println("改变后状态:"+originator.getState()); 
    originator.restoreMemento(caretaker.getMemento()); 
    System.out.println("恢复后状态:"+originator.getState()); 
  } 
} 

        代码演示了一个单状态单备份的例子,逻辑非常简单:Originator类中的state变量需要备份,以便在需要的时候恢复;Memento类中,也有一个state变量,用来存储Originator类中state变量的临时状态;而Caretaker类就是用来管理备忘录类的,用来向备忘录对象中写入状态或者取回状态。 

多状态多备份备忘录

       通用代码演示的例子中,Originator类只有一个state变量需要备份,而通常情况下,发起人角色通常是一个javaBean,对象中需要备份的变量不止一个,需要备份的状态也不止一个,这就是多状态多备份备忘录。实现备忘录的方法很多,备忘录模式有很多变形和处理方式,像通用代码那样的方式一般不会用到,多数情况下的备忘录模式,是多状态多备份的。其实实现多状态多备份也很简单,最常用的方法是,我们在Memento中增加一个Map容器来存储所有的状态,在Caretaker类中同样使用一个Map容器才存储所有的备份。下面我们给出一个多状态多备份的例子:

class Originator { 
  private String state1 = ""; 
  private String state2 = ""; 
  private String state3 = ""; 
  public String getState1() { 
    return state1; 
  } 
  public void setState1(String state1) { 
    this.state1 = state1; 
  } 
  public String getState2() { 
    return state2; 
  } 
  public void setState2(String state2) { 
    this.state2 = state2; 
  } 
  public String getState3() { 
    return state3; 
  } 
  public void setState3(String state3) { 
    this.state3 = state3; 
  } 
  public Memento createMemento(){ 
    return new Memento(BeanUtils.backupProp(this)); 
  } 
  public void restoreMemento(Memento memento){ 
    BeanUtils.restoreProp(this, memento.getStateMap()); 
  } 
  public String toString(){ 
    return "state1="+state1+"state2="+state2+"state3="+state3; 
  } 
} 
class Memento { 
  private Map<String, Object> stateMap; 
  public Memento(Map<String, Object> map){ 
    this.stateMap = map; 
  } 
  public Map<String, Object> getStateMap() { 
    return stateMap; 
  } 
  public void setStateMap(Map<String, Object> stateMap) { 
    this.stateMap = stateMap; 
  } 
} 
class BeanUtils { 
  public static Map<String, Object> backupProp(Object bean){ 
    Map<String, Object> result = new HashMap<String, Object>(); 
    try{ 
      BeanInfo beanInfo = Introspector.getBeanInfo(bean.getClass()); 
      PropertyDescriptor[] descriptors = beanInfo.getPropertyDescriptors(); 
      for(PropertyDescriptor des: descriptors){ 
        String fieldName = des.getName(); 
        Method getter = des.getReadMethod(); 
        Object fieldValue = getter.invoke(bean, new Object[]{}); 
        if(!fieldName.equalsIgnoreCase("class")){ 
          result.put(fieldName, fieldValue); 
        } 
      } 
    }catch(Exception e){ 
      e.printStackTrace(); 
    } 
    return result; 
  } 
  public static void restoreProp(Object bean, Map<String, Object> propMap){ 
    try { 
      BeanInfo beanInfo = Introspector.getBeanInfo(bean.getClass()); 
      PropertyDescriptor[] descriptors = beanInfo.getPropertyDescriptors(); 
      for(PropertyDescriptor des: descriptors){ 
        String fieldName = des.getName(); 
        if(propMap.containsKey(fieldName)){ 
          Method setter = des.getWriteMethod(); 
          setter.invoke(bean, new Object[]{propMap.get(fieldName)}); 
        } 
      } 
    } catch (Exception e) { 
      e.printStackTrace(); 
    } 
  } 
} 
class Caretaker { 
  private Map<String, Memento> memMap = new HashMap<String, Memento>(); 
  public Memento getMemento(String index){ 
    return memMap.get(index); 
  } 
  public void setMemento(String index, Memento memento){ 
    this.memMap.put(index, memento); 
  } 
} 
class Client { 
  public static void main(String[] args){ 
    Originator ori = new Originator(); 
    Caretaker caretaker = new Caretaker(); 
    ori.setState1("中国"); 
    ori.setState2("强盛"); 
    ori.setState3("繁荣"); 
    System.out.println("===初始化状态===\n"+ori); 
    caretaker.setMemento("001",ori.createMemento()); 
    ori.setState1("软件"); 
    ori.setState2("架构"); 
    ori.setState3("优秀"); 
    System.out.println("===修改后状态===\n"+ori); 
    ori.restoreMemento(caretaker.getMemento("001")); 
    System.out.println("===恢复后状态===\n"+ori); 
  } 
} 

备忘录模式的优缺点和适用场景

备忘录模式的优点有:

  • 当发起人角色中的状态改变时,有可能这是个错误的改变,我们使用备忘录模式就可以把这个错误的改变还原。
  • 备份的状态是保存在发起人角色之外的,这样,发起人角色就不需要对各个备份的状态进行管理。

备忘录模式的缺点:

  • 在实际应用中,备忘录模式都是多状态和多备份的,发起人角色的状态需要存储到备忘录对象中,对资源的消耗是比较严重的。

如果有需要提供回滚操作的需求,使用备忘录模式非常适合,比如jdbc的事务操作,文本编辑器的Ctrl+Z恢复等。

总结

以上所述是小编给大家介绍的Java设计模式之备忘录模式_动力节点Java学院,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对网站的支持!


# 备忘录模式  # java  # 设计模式  # Java设计模式之备忘录模式  # 深入理解Java设计模式之备忘录模式  # Java设计模式之java备忘录模式详解  # 实例讲解JAVA设计模式之备忘录模式  # 23种设计模式(18)java备忘录模式  # JAVA设计模式之备忘录模式原理与用法详解  # 详解备忘录模式及其在Java设计模式编程中的实现  # 详解Java设计模式之备忘录模式的使用  # Java设计模式之备忘录模式(Memento模式)介绍  # 类中  # 象中  # 便可  # 小编  # 就可以  # 都是  # 是一个  # 情况下  # 也有  # 有可能  # 在此  # 有很多  # 这就是  # 并在  # 也很  # 给大家  # 只有一个  # 要对  # 不需  # 这是个 


相关文章: 如何在Ubuntu系统下快速搭建WordPress个人网站?  香港服务器部署网站为何提示未备案?  如何在云服务器上快速搭建个人网站?  开源网站制作软件,开源网站什么意思?  制作营销网站公司,淘特是干什么用的?  打鱼网站制作软件,波克捕鱼官方号怎么注册?  手机网站制作平台,手机靓号代理商怎么制作属于自己的手机靓号网站?  网站制作公司排行榜,四大门户网站排名?  在线教育网站制作平台,山西立德教育官网?  网站好制作吗知乎,网站开发好学吗?有什么技巧?  股票网站制作软件,网上股票怎么开户?  javascript中对象的定义、使用以及对象和原型链操作小结  专业公司网站制作公司,用什么语言做企业网站比较好?  如何快速建站并高效导出源代码?  交易网站制作流程,我想开通一个网站,注册一个交易网址,需要那些手续?  建站之星伪静态规则如何设置?  猪八戒网站制作视频,开发一个猪八戒网站,大约需要多少?或者自己请程序员,需要什么程序员,多少程序员能完成?  内网网站制作软件,内网的网站如何发布到外网?  杭州银行网站设计制作流程,杭州银行怎么开通认证方式?  建站之星客服服务时间及联系方式如何?  制作国外网站的软件,国外有哪些比较优质的网站推荐?  SAX解析器是什么,它与DOM在处理大型XML文件时有何不同?  代购小票制作网站有哪些,购物小票的简要说明?  如何在云指建站中生成FTP站点?  如何通过建站之星自助学习解决操作问题?  专业企业网站设计制作公司,如何理解商贸企业的统一配送和分销网络建设?  如何使用Golang安装API文档生成工具_快速生成接口文档  网站专业制作公司,网站编辑是做什么的?好做吗?工作前景如何?  深圳网站制作的公司有哪些,dido官方网站?  如何在服务器上配置二级域名建站?  如何通过FTP服务器快速搭建网站?  成都品牌网站制作公司,成都营业执照年报网上怎么办理?  c# 在高并发场景下,委托和接口调用的性能对比  定制建站哪家更专业可靠?推荐榜单揭晓  网站制作的步骤包括,正确网址格式怎么写?  想学网站制作怎么学,建立一个网站要花费多少?  怎么制作网站设计模板图片,有电商商品详情页面的免费模板素材网站推荐吗?  关于BootStrap modal 在IOS9中不能弹出的解决方法(IOS 9 bootstrap modal ios 9 noticework)  香港服务器租用每月最低只需15元?  微信h5制作网站有哪些,免费微信H5页面制作工具?  孙琪峥织梦建站教程如何优化数据库安全?  建站之星安装提示数据库无法连接如何解决?  网站制作公司广州有几家,广州尚艺美发学校网站是多少?  C#怎么创建控制台应用 C# Console App项目创建方法  深圳企业网站制作设计,在深圳如何网上全流程注册公司?  如何实现建站之星域名转发设置?  c++23 std::expected怎么用 c++优雅处理函数错误返回【详解】  东莞市网站制作公司有哪些,东莞找工作用什么网站好?  专业网站建设制作报价,网页设计制作要考什么证?  如何通过山东自助建站平台快速注册域名? 

您的项目需求

*请认真填写需求信息,我们会在24小时内与您取得联系。