设计模式——备忘录模式

备忘录模式又称快照模式,是行为模式之一;

备忘录模式的应用场景是对某些对象做出了改变之后,又需要恢复到改变之前的状态!常常和命令模式结合使用...

备忘录中的三张角色;

1、原始角色,需要具有创建备忘录和根据备忘录恢复状态的方法

/*
 * Copyright (c) 2017. Xiaomi.Co.Ltd All rights reserved
 */

package com.pt.memo;

/**
 * @description 游戏角色
 * @author panteng
 * @date 17-2-24.
 */
public class Player {
    String name;    //角色名称
    Integer HP;     //血量
    Weapon weapon;  //武器

    public Player(){
    }

    public Player(String name, Integer HP, Weapon weapon){
        this.name = name;
        this.HP = HP;
        this.weapon = weapon;
    }
    public Memotor createMemotor(){
        //此处也并不是必须定义成final,定义成final这样可能会导致无法回收,但可以更有效的防止备忘录被修改
        final Memotor memotor = new Memotor(this.HP, this.weapon);
        return memotor;
    }

    /**
     * 深复制与浅复制的使用因场景而定
     * @return
     * @throws Exception
     */
    public Memotor createMemotorByDeepCopy() throws Exception{
        //采用深复制
        final Memotor memotor = new Memotor(new Integer(this.HP), (Weapon) this.weapon.deepClone());
        return memotor;
    }

    public void recover(Memotor memotor){
        this.HP = memotor.getHP();
        this.weapon = memotor.getWeapon();
    }

    public String getName(){
        return name;
    }
    public void setName(String name){
        this.name = name;
    }
    public Integer getHP(){
        return HP;
    }
    public void setHP(Integer HP){
        this.HP = HP;
    }
    public Weapon getWeapon(){
        return weapon;
    }
    public void setWeapon(Weapon weapon){
        this.weapon = weapon;
    }

    public void showState(){
        System.out.println(this.name + " HP:" + this.HP + " weapon:" + weapon);
    }
}

Player

武器类

/*
 * Copyright (c) 2017. Xiaomi.Co.Ltd All rights reserved
 */

package com.pt.memo;

import com.pt.prototype.PrototypeModle;

/**
 * @description 武器类
 * @author panteng
 * @date 17-2-24.
 */
public class Weapon extends PrototypeModle {
    String name;            //武器名称
    Float attackDistance;   //攻击距离
    Float ATK;              //攻击力
    public Weapon(){
    }
    public Weapon(String name, Float attackDistance, Float ATK){
        this.name = name;
        this.attackDistance = attackDistance;
        this.ATK = ATK;
    }

    public String getName(){
        return name;
    }
    public void setName(String name){
        this.name = name;
    }
    public Float getAttackDistance(){
        return attackDistance;
    }
    public void setAttackDistance(Float attackDistance){
        this.attackDistance = attackDistance;
    }
    public Float getATK(){
        return ATK;
    }
    public void setATK(Float ATK){
        this.ATK = ATK;
    }

    /**
     * Returns a string representation of the object. In general, the
     * {@code toString} method returns a string that
     * "textually represents" this object. The result should
     * be a concise but informative representation that is easy for a
     * person to read.
     * It is recommended that all subclasses override this method.
     * <p>
     * The {@code toString} method for class {@code Object}
     * returns a string consisting of the name of the class of which the
     * object is an instance, the at-sign character `{@code @}‘, and
     * the unsigned hexadecimal representation of the hash code of the
     * object. In other words, this method returns a string equal to the
     * value of:
     * <blockquote>
     * <pre>
     * getClass().getName() + ‘@‘ + Integer.toHexString(hashCode())
     * </pre></blockquote>
     *
     * @return a string representation of the object.
     */
    @Override
    public String toString(){
        return super.toString() + " name:" + this.name + " ATK:" + this.ATK + " attackDistance:" + attackDistance;
    }
}

Weapon

2、备忘录角色,用于保存原始角色需要记录的状态信息

/*
 * Copyright (c) 2017. Xiaomi.Co.Ltd All rights reserved
 */

package com.pt.memo;

/**
 * @description 游戏者备忘录
 * @author panteng
 * @date 17-2-24.
 */
public class Memotor {
    public Integer HP;
    public Weapon weapon;

    public Memotor(){
    }

    public Memotor(Integer HP, Weapon weapon){
        this.HP = HP;
        this.weapon = weapon;
    }

    public Integer getHP(){
        return HP;
    }
    public void setHP(Integer HP){
        this.HP = HP;
    }
    public Weapon getWeapon(){
        return weapon;
    }
    public void setWeapon(Weapon weapon){
        this.weapon = weapon;
    }
}

Memotor

3、备忘录管理者角色,主要做好封装,防止备忘录被修改

/*
 * Copyright (c) 2017. panteng.Co.Ltd All rights reserved
 */

package com.pt.memo;

/**
 * @description 备忘录管理者,之所以有此角色,是保护备忘录不被改变,备忘录模式使用上,除管理者和发起者外,其他不可以操作备忘录;
 * @author panteng
 * @date 17-2-24.
 */
public class MemotorManager {
    Memotor memotor;

    public MemotorManager(Memotor memotor){
        this.memotor = memotor;
    }

    public Memotor getMemotor(){
        return memotor;
    }
    public void setMemotor(Memotor memotor){
        this.memotor = memotor;
    }
}

4、测试类

/*
 * Copyright (c) 2017. Xiaomi.Co.Ltd All rights reserved
 */

package com.pt.memo;

import org.junit.Test;

/**
 * @description
 * @author panteng
 * @date 17-2-24.
 */
public class MemotorTest {
    @Test
    public void memotoeTest() throws Exception{
        //创建一个两个武器
        Weapon weapon1 = new Weapon("手枪", 200.1F, 100.5F);
        Weapon weapon2 = new Weapon("狙击枪", 1000.25F, 200.5F);
        //创建一个游戏角色
        Player player = new Player("魂斗罗", 1000, weapon1);
        player.showState();
        //战斗前先备份
        MemotorManager memotorManager = new MemotorManager(player.createMemotor());
        MemotorManager memotorManager1 = new MemotorManager(player.createMemotorByDeepCopy());
        System.out.println("战斗... ... 切换武器... ...");
        player.setWeapon(weapon2);  //切换武器
        player.setHP(0);            //挂了
        player.showState();
        //恢复
        System.out.println("恢复战斗前的状态... ...");
        player.recover(memotorManager.getMemotor());
        player.showState();
        //假如战斗过程中角色发生了这样的变化
        System.out.println("==========================================================");
        player.setHP(1);
        player.getWeapon().setATK(0F);
        player.getWeapon().setAttackDistance(0F);
        player.showState();
        player.recover(memotorManager.getMemotor());
        player.showState(); //武器无法恢复,因为备份过程中采用的是浅复制,血量恢复了,说明Integer采用的是深复制
        //使用第二种方式恢复
        player.recover(memotorManager1.getMemotor());
        player.showState();
    }
}
时间: 2024-10-13 19:48:22

设计模式——备忘录模式的相关文章

【C#设计模式-备忘录模式】

一.备忘录模式的定义: 在不破坏封闭的前提下,捕获一个对象的内部状态,并在该对象之外保存这个状态.这样以后就可将该对象恢复到原先保存的状态. 二.备忘录模式的结构和角色: 1.Originator(发起人):负责创建一个备忘录Memento,用以记录当前时刻自身的内部状态,并可使用备忘录恢复内部状态.Originator可以根据需要决定Memento存储自己的哪些内部状态. 2.Memento(备忘录):负责存储Originator对象的内部状态,并可以防止Originator以外的其他对象访问

我的设计模式:备忘录模式

我的设计模式-备忘录模式 恢复到最初的状态-->对原始状态的保留和恢复 联想到的业务:cms稿件放入到回收站,永久记录状态得用到数据库啊,怎么办?备忘录模式的内存缓存不适用? 类的单一职责 boy职责:状态   改变状态  设置状态  获取状态 备忘录职责:记录当前的状态   恢复状态 迪 米特原则:最少知道原则,不和陌生人说话 问题:备忘录只是记录了一个节点,多个备忘录记录了多个节点,可以恢复到任意节点?备忘录因该支持多个节点? 思考问题:增加备忘录管理者manager角色,有啥好处呢?没体验

[转] Android中的设计模式-备忘录模式

转自Android中的设计模式-备忘录模式 定义 备忘录设计模式的定义就是把对象的状态记录和管理委托给外界处理,用以维持自己的封闭性. 比较官方的定义 备忘录模式(Memento Pattern)又叫做快照模式(Snapshot Pattern)或Token模式,是GoF的23种设计模式之一,属于行为模式. 定义:在不破坏封闭的前提下,捕获一个对象的内部状态,并在该对象之外保存这个状态.这样以后就可将该对象恢复到原先保存的状态. 角色 笔记本:很多的内部状态需要被建立一个备忘录来管理,创建和取出

设计模式--备忘录模式(Memento)

什么是备忘录模式? 在不破坏封装性的前提下,捕获一个对象的内部状态,并在该对象之外保存这个状态,这样就可以将以后的对象状态恢复到先前保存的状态. 我们在编程的时候,经常需要保存对象的中间状态,当需要的时候,可以恢复到这个状态.比如,我们使用Eclipse进行编程时,假如编写失误(例如不小心误删除了几行代码),我们希望返回删除前的状态,便可以使用Ctrl+Z来进行返回.这时我们便可以使用备忘录模式来实现. 代码示例: 代码演示了一个单状态单备份的例子,逻辑非常简单:Originator类中的sta

[工作中的设计模式]备忘录模式memento

一.模式解析 备忘录对象是一个用来存储另外一个对象内部状态的快照的对象.备忘录模式的用意是在不破坏封装的条件下,将一个对象的状态捕捉(Capture)住,并外部化,存储起来,从而可以在将来合适的时候把这个对象还原到存储起来的状态.备忘录模式常常与命令模式和迭代子模式一同使用. 备忘录模式可以根据客户指令,将相应的对象特有属性进行快照,如果客户要恢复对象,则根据快照提供的特有属性进行还原. 二.模式代码 package memento.patten; /** *备忘录类,同时指定要保存的对象属性

PHP设计模式——备忘录模式

声明:本系列博客參考资料<大话设计模式>,作者程杰. 备忘录模式又叫做快照模式或Token模式,在不破坏封闭的前提下.捕获一个对象的内部状态,并在该对象之外保存这个状态.这样以后就可将该对象恢复到原先保存的状态. UML类图: 角色: 1.发起人(GameRole):负责创建一个备忘录,用以记录当前时刻自身的内部状态,并可使用备忘录恢复内部状态.发起人能够依据须要决定备忘录存储自己的哪些内部状态. 2.备忘录(RoleStateSaveBox):负责存储发起人对象的内部状态,并能够防止发起人以

设计模式-备忘录模式实现悔棋操作

利用设计模式中的备忘录模式实现多步悔棋的操作 1 import java.util.*; 2 class Chessman { 3 private String label; 4 private int x; 5 private int y; 6 public static int index=-1; 7 public Chessman(String label,int x,int y) { 8 this.label = label; 9 this.x = x; 10 this.y = y; 1

[设计模式] 备忘录模式Memento Pattern

在GOF的<设计模式:可复用面向对象软件的基础>一书中对备忘录模式是这样说的:在不破坏封装性的前提下,捕获一个对象的内部状态,并在该对象之外保存这个状态.这样以后就可将该对象恢复到原先保存的状态. 类图和实例: 简单的模式实例: #include <iostream> #include <string> using namespace std; class Memento { private:     string state; public:     Memento(

大话设计模式—备忘录模式

备忘录模式(Memento Pattern)保存一个对象的某个状态,以便在适当的时候恢复对象.备忘录模式属于行为型模式.所谓备忘录模式就是在不破坏封装的前提下,捕获一个对象的内部状态,并在该对象之外保存这个状态,这样可以在以后将对象恢复到原先保存的状态.很多时候我们总是需要记录一个对象的内部状态,这样做的目的就是为了允许用户取消不确定或者错误的操作,能够恢复到他原先的状态,使得他有"后悔药"可吃. 大话设计模式中程杰老师给出的定义是,备忘录模式:在不破坏封装性的前提下,捕获一个对象的内

深入浅出设计模式——备忘录模式(Memento Pattern)

模式动机 为了使软件的使用更加人性化,对于误操作,我们需要提供一种类似“后悔药”的机制,让软件系统可以回到误操作前的状态,因此需要保存用户每一次操作时系统的状态,一旦出现误操作,可以把存储的历史状态取出即可回到之前的状态.现在大多数软件都有撤销(Undo)的功能,快捷键一般都是Ctrl+Z,目的就是为了解决这个后悔的问题. 在应用软件的开发过程中,很多时候我们都需要记录一个对象的内部状态.在具体实现过程中,为了允许用户取消不确定的操作或从错误中恢复过来,需要实现备份点和撤销机制,而要实现这些机制