【设计模式】你知道游戏SL大法是什么设计模式吗?

什么是备忘录模式?

老规矩,我们先来看看备忘录模式 (Memento) 的定义:在不破坏封装性的前提下,捕获一个对象的内部状态,并在该对象之外保存这个状态。这样以后就可将该对象恢复到原先保存的状态。

它的UML类图如下:

从上图可以发现,备忘录模式其实整体分为三个部分:Originator(发起人)、备忘录(Memento)、CareTaker(管理者)。其中,Originator负责创建一个备忘录Memento,并哟个memento记录当前时刻它的内部状态 (部分或整体) ,同时,也可以使用备忘录Memento恢复到某一个状态;备忘录Memento负责记录、存储发起人Originator的内部状态,并且只有发起人Originator能够访问其内部数据;而管理者Caretaker就负责保存备忘录Memento,注意,它不会查看备忘录里的具体内容。

举一个大家熟知的例子,SL大法知道吧?

在游戏里,当我们保存进度时,玩家角色就是一个发起人,而角色的当前状态(等级、血量、魔法值等)被保存(Save);当我们想重来的时候,读取存档(Load),就能够恢复角色到之前某一个时刻的状态。上述整个过程其实就是备忘录模式的具体体现,接下来我们就以游戏保存、读取来看看备忘录模式的具体应用。

如何使用备忘录模式?

先定义一个英雄类,表示玩家操作的游戏角色,他拥有生命值、魔法值、攻击力、防御力:

  • 英雄类
@Data
public class Hero {
    private int hp;
    private int mp;
    private int atk;
    private int def;

    // 初始角色状态
    public void getInitialState() {
        this.hp = 100;
        this.mp = 50;
        this.atk = 10;
        this.def = 5;
    }

    // 当前状态展示
    public void displayState() {
        System.out.println("角色当前状态:");
        System.out.println("生命值:" + this.hp);
        System.out.println("魔法值:" + this.hp);
        System.out.println("攻击力:" + this.atk);
        System.out.println("防御力: " + this.def);
        System.out.println("******************");
    }

    // 角色经验升级,恢复一定生命、魔法并增加属性
    public void levelUp() {
        this.hp += 20;
        this.mp += 10;
        this.atk += 5;
        this.def += 5;
    }

    // 大战boss
    public void fight() {
        this.hp -= 99;
        this.mp -= 50;
    }

    // 保存现在状态
    public SavaMemento saveState() {
        return (new SavaMemento(hp, mp, atk, def));
    }

    // 恢复之前状态
    public void loadState(SavaMemento memento) {
        this.hp = memento.getHp();
        this.mp = memento.getMp();
        this.atk = memento.getAtk();
        this.atk = memento.getDef();
    }
}

我们注意到,英雄类里除了初始化、战斗等函数,有两个公共方法:SaveMemento和loadState,前者返回了一个存档,后者则根据这个存档恢复到某一个时刻的状态。其实也可以这么理解,保存存档的时候,得到了一个存档id,之后读取存档则需要该存档id。

  • 存档类

存档类的实现很简单,就是根据最初系统的设计目的,来保存对应的英雄类中的某些信息。

public class SavaMemento {
    private int hp; // 生命力
    private int mp; // 魔法值
    private int atk; // 攻击力
    private int def; // 防御力

    public SavaMemento(int hp, int mp, int atk, int def) {
        this.hp = hp;
        this.mp = hp;
        this.atk = atk;
        this.def = def;

    }
}
  • 存档保管箱

存档保存角色信息,而存档保管箱则是将存档的实现细节收敛,防止客户端知道其具体的实现细节。

public class CareTaker {

    private SavaMemento savaLoadMemento;

    public SavaMemento loadGame() {
        try {
            if (savaLoadMemento != null)
                return savaLoadMemento;
            else
                throw new Exception();
        } catch (Exception ignored) {
            System.out.println("The saveFile is missed!");
        }
        return new SavaMemento(100, 50, 10, 5);
    }

    public void saveGame(SavaMemento memento) {
        this.savaLoadMemento = memento;
    }
}
  • 游戏客户端

设计一个游戏客户端,操作一个游戏角色从出生到大战boss、再到读取存档的过程:

public class GameClient {

    public static void main(String[] args) {

        // 显示初始状态
        System.out.println("初始状态--");
        Hero hero = new Hero();
        hero.getInitialState();
        hero.displayState();

        // 保存游戏
        System.out.println("保存游戏--");
        CareTaker caretaker = new CareTaker();
        caretaker.saveGame(hero.saveState());

        // 打死大boss
        System.out.println("三拳打死Boss--");
        hero.fight();
        hero.displayState();

        // 升级啦
        System.out.println("经验升级--");
        hero.levelUp();
        hero.displayState();

        // 恢复游戏
        System.out.println("读取存档--");
        hero.loadState(caretaker.loadGame());
        hero.displayState();
    }
}
  • 来看看输出

总结

应用场景

备忘录模式主要应用在这样的场景:需要记录某一个对象的内部状态,并且能够将对象恢复到某一个时刻状态,例如用户操作redo、或者游戏进度保存等等。

优点

提供了一种可以恢复状态的机制,进行回溯甚至将对象恢复到任意时刻的状态。

缺点

会额外消耗资源,因为需要将对象的状态信息进行保存,不过这个缺点在实际业务场景下都是可以接受的。

参考资料

《大话设计模式》

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/337274.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Django(八)

1. 管理员操作 1.1 添加 from django.shortcuts import render, redirectfrom app01 import models from app01.utils.pagination import Paginationfrom django import forms from django.core.exceptions import ValidationError from app01.utils.bootstrap import BootStr…

【MySQL】where和having的区别

🍎个人博客:个人主页 🏆个人专栏:数据库 ⛳️ 功不唐捐,玉汝于成 目录 前言 正文 用途: 使用位置: 操作对象: 聚合函数: 示例: 结语 我的其他博客 前言 数据库中的 WHERE 和 HAVING 子句在 SQL 查…

排序算法整理

快速排序 C实现 void fastStore(int *a, int start, int end){if(start>end)return ;int leftstart;int rightend;int tempa[left];//设置基准值tempwhile(left < right) //左指针的位置一定小于右指针的位置{while(a[right]>temp && left < right) //左…

RabbitMQ-消息延迟

一、死信交换机 1、描述 一个队列接收到的消息有过期时间&#xff0c;消息过期之后&#xff0c;如果配置有死信队列&#xff0c;消息就会进去死信队列。 2、图解 3、过程 当生产者将消息发送到exchange1&#xff0c;然后交换机将消息路由到队列queue1&#xff0c;但是队列que…

快速入门:使用 Gemini Embeddings 和 Elasticsearch 进行向量搜索

Gemini 是 Google DeepMind 开发的多模态大语言模型家族&#xff0c;作为 LaMDA 和 PaLM 2 的后继者。由 Gemini Ultra、Gemini Pro 和 Gemini Nano 组成&#xff0c;于 2023 年 12 月 6 日发布&#xff0c;定位为 OpenAI 的竞争者 GPT-4。 本教程演示如何使用 Gemini API 创建…

浅析Java中volatile关键字

认识volatile关键字 Java中的volatile关键字用于修饰一个变量&#xff0c;当这个变量被多个线程共享时&#xff0c;这个变量的值如果发生更新&#xff0c;每个线程都能获取到最新的值。volatile关键字在多线程环境下还会禁止指令重排序&#xff0c;确保变量的赋值操作按照代码的…

书生·浦语大模型实战营-学习笔记4

XTuner 大模型单卡低成本微调实战 Finetune简介 常见的两种微调策略&#xff1a;增量预训练、指令跟随 指令跟随微调 数据是一问一答的形式 对话模板构建 每个开源模型使用的对话模板都不相同 指令微调原理&#xff1a; 由于只有答案部分是我们期望模型来进行回答的内容…

敏捷测试和DevOpes自动化测试的区别

敏捷测试和DevOps自动化测试在以下方面存在区别&#x1f447; 1️⃣目标 &#x1f388;敏捷测试的主要目标是提供快速的反馈和持续的改进&#xff0c;以便在开发过程中尽早发现和解决问题&#xff0c;从而提高软件的质量和可靠性。 &#x1f308;DevOps自动化测试的目标是提高软…

C语言算法赛——蓝桥杯(省赛试题)

一、十四届C/C程序设计C组试题 十四届程序C组试题A#include <stdio.h> int main() {long long sum 0;int n 20230408;int i 0;// 累加从1到n的所有整数for (i 1; i < n; i){sum i;}// 输出结果printf("%lld\n", sum);return 0; }//十四届程序C组试题B…

统计学-R语言-7.1

文章目录 前言假设检验的原理假设检验的原理提出假设做出决策表述结果效应量 总体均值的检验总体均值的检验(一个总体均值的检验) 练习 前言 本章主题是假设检验(hypothesis testing)。与参数估计一样&#xff0c;假设检验也是对总体参数感兴趣&#xff0c;如比例、比例间的差…

负载均衡流程

1、负载均衡流程图 2、触发负载均衡函数trigger_load_balance void trigger_load_balance(struct rq *rq) { /* Dont need to rebalance while attached to NULL domain */ if (unlikely(on_null_domain(rq)))//当前调度队列中的调度域是空的则返回 return; i…

SpringMVC数据校验

导包 配置springmvc.xml <bean id"validator" class" org.springframework.validation.beanvalidation.LocalValidatorFactoryBean"><property name"providerClass" value"org.hibernate.validator.HibernateValidator ">…

『C++成长记』模板

&#x1f525;博客主页&#xff1a;小王又困了 &#x1f4da;系列专栏&#xff1a;C &#x1f31f;人之为学&#xff0c;不日近则日退 ❤️感谢大家点赞&#x1f44d;收藏⭐评论✍️ 目录 一、泛型编程 二、函数模板 &#x1f4d2;2.1函数模板概念 &#x1f4d2;2.2函数…

羊驼系列大模型LLaMa、Alpaca、Vicuna

羊驼系列大模型&#xff1a;大模型的安卓系统 GPT系列&#xff1a;类比ios系统&#xff0c;不开源 LLaMa让大模型平民化 LLaMa优势 用到的数据&#xff1a;大部分英语、西班牙语&#xff0c;少中文 模型下载地址 https://huggingface.co/meta-llama Alpaca模型 Alpaca是斯…

ELK之Grafana读取ES-Nginx数据后地图不展示的问题

一、背景&#xff1a;Grafana读取ES-Nginx数据后地图不展示 二、解决方法 [rootsg grafana-worldmap-panel]# pwd /var/lib/grafana/plugins/grafana-worldmap-panel [rootsg grafana-worldmap-panel]# [rootsg grafana-worldmap-panel]# sed -i s/https:\/\/cartodb-basemap…

基于SpringBoot Vue高校失物招领系统

大家好✌&#xff01;我是Dwzun。很高兴你能来阅读我&#xff0c;我会陆续更新Java后端、前端、数据库、项目案例等相关知识点总结&#xff0c;还为大家分享优质的实战项目&#xff0c;本人在Java项目开发领域有多年的经验&#xff0c;陆续会更新更多优质的Java实战项目&#x…

Python-基础篇-数据结构-列表、元组、字典、集合

文章目录 思维导图❓ 大抵是何物数据结构切片 &#x1f4ac;具体是何物列表&#x1f4bb; list&#x1f4bb; [ ]自我介绍精神面貌使用说明生理体征增删查改 方法汇总 元组&#x1f4bb; tuple&#x1f4bb; ( )自我介绍使用说明精神面貌生理体征增删查改 字典&#x1f4bb; di…

有什么提高编程能力的书籍推荐吗?

数据密集型应用系统设计 原文完整版PDF&#xff1a;https://pan.quark.cn/s/d5a34151fee9 这本书的作者是少有的从工业界干到学术界的牛人&#xff0c;知识面广得惊人&#xff0c;也善于举一反三&#xff0c;知识之间互相关联&#xff0c;比如有个地方把读路径比作programming …

qt学习:QT对话框+颜色+文件+字体+输入

目录 概述 继承图 QColorDialog 颜色对话框 QFileDialog 文件对话框 保存文件对话框 QFontDialog 字体对话框 QInputDialog 输入对话框 概述 对于对话框的功能&#xff0c;在GUI图形界面开发过程&#xff0c;使用是非常多&#xff0c;那么Qt也提供了丰富的对话框类QDia…

R2DBC-响应式数据库

简单查询 基于全异步,响应式,消息驱动 用法: 1.导入驱动:导入连接池(r2dbc-pool),导入驱动(r2dbc-mysql) 2. 使用驱动提供的api操作 pom.xml <properties><r2dbc-mysql.version>1.0.5</r2dbc-mysql.version> </properties><dependencies><d…