首页 > 其他分享 >备忘录模式

备忘录模式

时间:2023-10-09 23:33:21浏览次数:54  
标签:状态 int defensePower attackPower 模式 备忘录 public

备忘录模式

案例引入

游戏角色状态恢复问题

游戏角色有攻击力,防御力等,在大战BOSS前保存自身的状态(攻击力,防御力),当大战BOSS后攻击力和防御力下降,从备忘录对象恢复到大站前的状态。

传统方式实现案例

创建一个游戏角色类对应的状态类,给每个游戏角色对象,对应一个对应状态类的对象,用来保存状态。

传统方式实现问题分析
  • 1.一个游戏对象,就对应一个保存游戏状态的对象,这样如果游戏中有很多游戏角色,也会产生很多与之对应的状态角色,不利于管理,其内存开销很大。
  • 2.传统的方式,就是简单的备份,new一个另外的对象来,再把需要备份的数据放到这个新对象,但这就暴露了对象内部的细节。
  • 3.解决方案 => 备忘录模式。

基本介绍

  • 1.备忘录模式(Memento Pattern)在不破坏封装性的前提下,捕获了一个对象的内部状态,并在该对象之外保存了该状态。这样需要恢复到原有状态时,就可将该对象恢复到原先保存时的状态。
  • 2.也可以这样理解备忘录模式,现实生活中的备忘录是用来记录某些要去做的事情,或者是记录已经达成的共同意见的事情,以防忘记了。在软件层面,备忘录也有着相同的含义,备忘录对象主要用来记录一个对象的某种状态,或者数据,当要做回退时,可用从备忘录对象里获取原有的状态数据进行恢复。
  • 3.备忘录模式属于行为型模式。

备忘录模式原理类图

角色分析
  • 1.Originator是原始类,代表需要保存状态和数据信息的类。
  • 2.Memento是备忘录类,是保存Originator类对象状态和数据信息的类。
  • 3.Caretaker是一个管理备忘录的类,其中有一个保存备忘录对象的集合属性。
  • 4.说明,如果希望保存多个Originator对象的不同时间的状态,只需要用HashMap<String,Map<String,Object>>保存就可以了
原理的代码
/**
 * @author 长名06
 * @version 1.0
 * 保存对象的状态/数据
 */
public class Memento {
    //保存的状态
    private String savedState;

    public Memento(String s){
        savedState = s;
    }

    public String getStateInfo(){//获取状态信息
        return savedState;
    }
}
/**
 * @author 长名06
 * @version 1.0
 * 游戏角色类 需要保存状态的类
 */
public class Originator {

    //状态信息
    private String state;

    public String getState() {
        return state;
    }

    public void setState(String state) {
        this.state = state;
    }

    //可以保存状态信息到备忘录对象中,并返回该备忘录对象
    public Memento saveStateMemento(){
        return new Memento(state);
    }

    //通过备忘录对象,恢复对象状态
    public void restoreStateFromMemento(Memento memento){
        state = memento.getStateInfo();
    }
}
/**
 * @author 长名06
 * @version 1.0
 * 管理所有的备忘录对象
 */
public class Caretaker {
    private List<Memento> mementoList = new ArrayList<>();

    public void add(Memento memento){
        mementoList.add(memento);
    }

    /**
     * 根据索引获取 原始对象 对应的 备忘录对象
     * @param index
     * @return
     */
    public Memento get(int index){
        return mementoList.get(index);
    }

    public void delete(Memento memento){
        mementoList.remove(memento);
    }
}
public class Client {
    public static void main(String[] args) {
        Originator originator = new Originator();
        Caretaker caretaker = new Caretaker();
        originator.setState(" 状态#1 攻击力是100 ");
        //保存了状态1
        caretaker.add(originator.saveStateMemento());
        //保存了状态2
        originator.setState(" 状态#2 攻击力是80 ");
        caretaker.add(originator.saveStateMemento());
        //保存了状态3
        originator.setState(" 状态#2 攻击力是60 ");
        caretaker.add(originator.saveStateMemento());

        System.out.println("当前状态" + originator.getState());
        //恢复到状态1
        originator.restoreStateFromMemento(caretaker.get(0));
        System.out.println("恢复后的装态" + originator.getState());
    }
}

备忘录模式实现案例

案例要求

游戏角色有攻击力和防御力,在打BOSS前保存自身的状态(攻击力和防御力),当大战BOSS后攻击力和防御力下降,从备忘录对象恢复到大战前的状态。

类图

代码
/**
 * @author 长名06
 * @version 1.0
 * 游戏角色的备忘录类
 */
public class Mementor {

    private int attackPower;//攻击力

    private int defensePower;//防御力

    public Mementor(int attackPower,int defensePower){
        this.attackPower = attackPower;
        this.defensePower = defensePower;
    }

    public int getAttackPower() {
        return attackPower;
    }

    public void setAttackPower(int attackPower) {
        this.attackPower = attackPower;
    }

    public int getDefensePower() {
        return defensePower;
    }

    public void setDefensePower(int defensePower) {
        this.defensePower = defensePower;
    }
}
/**
 * @author 长名06
 * @version 1.0
 * 游戏角色
 */
public class GameRole {

    private int attackPower;

    private int defensePower;


    public Mementor createMementor(){
        return new Mementor(attackPower,defensePower);
    }

    public void restoreStateFromMementor(Mementor mementor){
        attackPower = mementor.getAttackPower();
        defensePower = mementor.getDefensePower();
    }

    public void showState(){
        System.out.println("游戏角色当前攻击力" + this.attackPower + "防御力" + this.defensePower);
    }

    public int getAttackPower() {
        return attackPower;
    }

    public void setAttackPower(int attackPower) {
        this.attackPower = attackPower;
    }

    public int getDefensePower() {
        return defensePower;
    }

    public void setDefensePower(int defensePower) {
        this.defensePower = defensePower;
    }
}
/**
 * @author 长名06
 * @version 1.0
 * 守护者对象,用于管理(保存游戏状态/数据对象)的对象
 */
public class Caretaker {

    //只保存一个角色的一次状态
    private Mementor mementor;

    //对于一个游戏角色的多个状态
//    private List<Mementor> mementorList = new ArrayList<>();

    //保存多个游戏角色的状态
//    private HashMap<String,ArrayList<Mementor>> rolesMap;


    public Mementor getMementor() {
        return mementor;
    }

    public void setMementor(Mementor mementor) {
        this.mementor = mementor;
    }
}
public class Client {
    public static void main(String[] args) {
        GameRole gameRole = new GameRole();
        gameRole.setAttackPower(100);
        gameRole.setDefensePower(200);
        System.out.print("打BOSS前状态\t");
        gameRole.showState();

        //将当前状态保存到一个备忘录类中
        Caretaker caretaker = new Caretaker();
        caretaker.setMementor(gameRole.createMementor());

        //打BOSS状态下降
        gameRole.setDefensePower(100);
        gameRole.setAttackPower(50);
        //打BOSS后状态
        System.out.print("打BOSS后状态\t");
        gameRole.showState();

        //恢复状态
        gameRole.restoreStateFromMementor(caretaker.getMementor());
        //展示恢复后的状态
        System.out.print("恢复后的状态\t");
        gameRole.showState();
    }
}

注意事项和细节

  • 1.给用户提供了一种可以恢复状态的机制,可以使用户能够比较方便的回到某个历史的状态。
  • 2.实现了信息的封装,使得用户不需要关心状态的保存细节。
  • 3.如果类的成员变量过多,势必会占用比较多的内存资源,而且每次保存都会消耗内存。
  • 4.适用的应用场景:4.1 后悔药;4.2 打游戏时的存档;4.3 Windows里的ctrl + z; 4.4 浏览器里的后退; 4.5 数据库的事务撤销。
  • 5.为了节约内存,备忘录模式可以和原型模式配合使用。
    只是为了记录自己的学习历程,且本人水平有限,不对之处,请指正。

标签:状态,int,defensePower,attackPower,模式,备忘录,public
From: https://www.cnblogs.com/changming06/p/17753446.html

相关文章

  • 【愚公系列】2023年10月 二十三种设计模式(八)-组合模式(Composite Pattern)
    ......
  • 笔记1:环境安装及烧录模式
    1.需要安装ADB工具2.使用RKDevTool.exe 烧录固件 K3568开发板需要进入Loader或Maskrom模式才可执行烧写操作。进入Loader模式的方法:首先按住开发板上的音量+(V+)按键(具体位置请参考按键示意图3.2.3)不松,给开发板上电或复位,此时RKDevTool工具会提示:发现一个LOADER......
  • 设计模式分类(背)
           ......
  • Bridge 桥接模式简介与 C# 示例【结构型2】【设计模式来了_7】
    〇、简介1、什么是桥接模式?一句话解释:  通过一个类的抽象,与另一个类的抽象关联起来,当做桥。此后不管两个抽象类的实现有多少种,均可以通过这个桥来将两个对象联系起来。桥接,顾名思义就是用桥来连接河两岸,将原本不关联的两部分联系起来,且不影响两岸的各自演化,演化出来的不同对......
  • Java设计模式之适配器模式
    1.1.概述如果去欧洲国家去旅游的话,他们的插座如下图最左边,是欧洲标准。而我们使用的插头如下图最右边的。因此我们的笔记本电脑,手机在当地不能直接充电。所以就需要一个插座转换器,转换器第1面插入当地的插座,第2面供我们充电,这样使得我们的插头在当地能使用。生活中这样的例子很......
  • 工厂设计模式
      Spring底层框架就用到工厂方法模式 ......
  • 责任链设计模式
    过滤器   ......
  • 学习笔记-设计模式-创建型模式-单例模式
    单例模式一个类只有一个实例,并提供一个全局访问此实例的点,哪怕多线程同时访问。单例模式主要解决了一个全局使用的类被频繁的创建和消费的问题。单例模式的案例场景数据库的连接池不会反复创建spring中一个单例模式bean的生成和使用在我们平常的代码中需要设置全局的一些属......
  • Java设计模式之建造者模式
    1.1.概述将一个复杂对象的构建与表示分离,使得同样的构建过程可以创建不同的表示。• 分离了部件的构造(由Builder来负责)和装配(由Director负责)。从而可以构造出复杂的对象。这个模式适用于:某个对象的构建过程复杂的情况。• 由于实现了构建和装配的解耦。不同的构建器,相......
  • 设计模式 (1): 5 种创建型模式 (结合代码详解)
    目录1单例模式饿汉单例懒汉单例双重检验单例静态内部类单例破坏单例(反射、反序列化)枚举类单例2工厂模式简单工厂模式工厂方法模式3抽象工厂模式代码实现对比三种工厂模式如何选择工厂方法和抽象工厂?4建造者模式5原型模式1单例模式需考虑的问题:是否线程安全是否延......