设计模式与游戏完美开发(3)

更多内容可以浏览本人博客:https://azureblog.cn/ 😊
该文章主体内容来自《设计模式与游戏完美开发》—蔡升达

第二篇 基础系统

第五章 获取游戏服务的唯一对象——单例模式(Singleton)

游戏实现中的唯一对象

在游戏开发过程中,我们常常希望一些游戏对象(比如某游戏功能管理器)具有两项特性:

  • 唯一性:同时间只存在一个对象。
  • 便捷性:提供一个快速获取这个对象的方法。

比较直接的想法是使用全局静态对象,但是全局静态对象很难避免产生多个对象,也容易产生全局变量名重复的问题。

所以最好让这个类只产生一个对象,并提供便利的方法来获取这唯一的对象,这就是单例模式

单例模式

确认类只有一个对象,并提供一个全局的方法来获取这个对象。——GoF

结构:

img

参与者如下:

  • 能产生唯一对象的类,并且提供"全局方法"让外界可以方便获取唯一的对象

  • 通常会把唯一的类对象设置为"静态类属性"

  • 习惯上会使用Instance作为全局静态方法的名称,通过这个静态函数可能获取"静态类属性"

C#实现范例:

public class Singleton {public string Name { get; set; }private static Singleton _instance;public static Singleton Instance {get {if (_instance == null) {//保证唯一性,并提供遍历方法供使用Debug.Log("产生Singleton");_instance = new Singleton();}return _instance;}}private Single() { }//让外部不能new对象,保证唯一性
}void UnitTest() {Singleton.Instance.Name = "Hello";Singleton.Instance.Name = "World";Debug.Log(Singleton.Instance.Name);
}

使用单例模式获取唯一的游戏服务对象

一般来说,游戏需要一个GameManger类来管理一些功能和全局数据,在《P级阵地》中,PBaseDefenseGame就代表这样一个类,并且应用单例模式设计这个类:

img

参与者说明:

  • PBaseDefenseGame

    • 游戏主程序,内部包含了类型为PBaseDefenseGame的静态成员属性_instance,作为该类唯一的对象。
    • 提供使用C# getter实现的静态成员方法Instance,用它来获取唯一的静态成员属性_instance
  • BattleScene

    • PBaseDefenseGame类的客户端,使用PBaseDefenseGame.Instance来获取唯一的对象
PBaseDefenseGame.cspublic class PBaseDefenseGame {private static PBaseDefenseGame _instance;public static PBaseDefenseGame Instance {get {if (_instance == null) {_instance = new PBaseDefenseGame();}return _instance;}}...private PBaseDefenseGame() { }
}BattleState.cspublic class BattleScene: ISceneState {...pubic override void StateBegin() {PBaseDefenseGame.Instance.Initinal();//通过Instance直接访问PBaseDefenseGame对象}...
}SoldierClickScript.cspublic class SoldierOnClick: MonoBehavior {...public void OnClick() {PBaseDefenseGame.Instance.ShowSoldierInfo(Solder);}
}

反对使用单例模式的原因

单例模式似乎看起来非常方便,不必为了“安排参数传递”和“设置引用”伤脑筋,你可以在项目的任何地方通过Instance方法使用该类对象。然而,大多数资深设计者都反对滥用单例模式,有以下几个原因:

  • 全局变量的过度滥用:实质上,单例模式类也是一种全局变量,然而绝大多数类从设计角度需要保有“适当可视性”,很多类我们并不需要甚至并不愿意它被全局共享,单例模式在这种情况下可以被认为是一种“放弃思考”的暴力求解手段。
  • 违反“开-闭原则”:让一个类成为单例类,理论上它就失去了继承能力,就无法对修改关闭。

但是,也有两种方法可以让单例模式返回接口类——即父类为单例模式,并让子类继承实现:

  • 子类向父类注册实体对象,让父类的Instance方法返回对象时,按条件查表发挥对应的子类对象。
  • 每个子类都实现单例模式,再由父类的Instance去获取这些子类。(《P级阵地》采用类似的方法)。

少用单例模式如何方便地引用到单一对象

让类具有计数功能,即通过一个静态类属性计数器来限制对象的数量。

**将A类对象设置为B类的成员,B类的方法便可以方便的使用A类对象。这也是“依赖注入”的方式之一。**可以让被引用的对象不必通过参数传递的方式,就能被类的其他方法引用.按照设置的方式又可以分为"分别设置"和"指定类静态成员"两种。

  1. 分别设置

    在初始化的时候给用到的每个B类将A类对象传入

    public class PBaseDefenseGame {public void Initinal() {m_GameEventSystem = new GameEventSystem(this);//将PBaseDefenseGame类对象传入...}
    }public abstract class IGameSystem {protected PBaseDefenseGame m_PBDGame = null;public IGameSystem(PBaseDefenseGame PBDGame) {m_PBDGame = PBDGame;}
    }public class CampSystem: IGameSystem {public CampSystem(PBaseDefenseGame PBDGame): base(PBDGame) {Initialize();}public void ShowCaptiveCamp() {m_PBDGame.ShowGameMsg("获得俘兵营");//每个系统都可以通过成员m_PBDGame使用PBaseDefenseGame类对象的方法和属性}
    }
    
  2. 指定类的静态成员

    A类的功能若需要使用到B类的方法,并且A类在产生其对象时具有下列几种情况:

    • 产生对象的位置不确定

    • 有多个地方可以产生对象

    • 生成的位置无法引用到

    • 有众多子类

      (实际上与上面相对,1.分别设置中的产生对象位置就是确定的)

​ 当满足上述情况之一时,可以直接将B类对象设置为A类中的"静态成员属性", 让该类的对象都可以直接使用。

// PBaseDefenseGame.cspublic class PBaseDefenseGame {public void Initinal() {m_StageSystem = new StageSystem(this);// 注入其他系统EnemyAI.SetStageSystem(m_StageSystem);}
}
//举例来说,敌方单位AI类(EnemyAI), 在运行时需要使用关卡系统(StageSystem)的信息,但EnemyAI对象产生的位置是在敌方单位建造者(EnemyBuilder)之下:
EnemyBuilder.cspublic class EnemyBuilder: ICharacterBuilder {public override void AddAI() {EnemyAI theAI = new EnemyAI(m_BuildParam.NewCharacter, m_BuildParam.AttackPosition);m_BuildParam.NewCharacter.SetAI(theAI);}
}
//按照"最少知识原则(LKP)",会希望敌方单位的建造者(EnemyBuilder)减少对其他无关类的引用.因此,在产生敌方单位AI(EnemyAI)对象时,敌方单位建造者(EnemyBuilder)无法将关卡系统(StageSystem)对象设置给敌方单位AI,这是属于上述"生成的位置无法引用到"的情况.所以,可以在敌方单位AI(EnemyAI)类中,提供一个静态成员属性和静态方法,让关卡系统(StageSystem)对象产生的当下,就设置给敌方单位AI(EnemyAI)类:
public class EnemyAI: ICharacterAI {private static StageSystem m_StageSystem = null;public static void SetStageSystem(StageSystem StageSystem) {m_StageSystem = StageSystem;}public ovrride bool CanAttackHeart() {m_StageSystem.LoseHeart();return true;}
}
  1. 直接使用静态类
public static class PBDFactory {private static IAssetFactory m_AssetFactory = null;public static IAssetFactory GetAssetFactory() {if (m_AssetFactory == null) {if (m_bLoadFromResource) {m_AssetFactory = new ResourceAssetFactory();} else {m_AssetFactory = new RemoteAssetFactory();}}return m_AssetFactory;}
}

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【工业场景】用YOLOv8实现工业安全帽识别

工业安全帽识别是一项重要的工作安全管理措施,旨在防止工作场所发生头部伤害事故。通过使用YOLOv8等深度学习模型,可以实时准确地检测出工人是否佩戴安全帽,及时发现违规行为,为工人提供更安全的工作环境。 使用YOLOv8实现工业安全…

路由器的转发表

【4-24】 已知路由器R₁ 的转发表如表T-4-24 所示。 表T-4-24 习题4-24中路由器R₁的转发表 前缀匹配 下一跳地址 路由器接口 140.5.12.64/26 180.15.2.5 m2 130.5.8/24 190.16.6.2 ml 110.71/16 ----- m0 180.15/16 ----- m2 190.16/16 ----- ml 默认 11…

打开idea开发软件停留在加载弹出框页面进不去

问题 idea软件点击打开,软件卡在加载弹框进不去。 解决方法 先进入“任务管理器”停止IDEA的任务进程 2.找到IDEA软件保存的本地数据文件夹 路径都是在C盘下面:路径:C:\Users\你的用户名\AppData\Local\JetBrains 删除目录下的文件夹&…

【网络安全 | 漏洞挖掘】HubSpot 全账户接管(万字详析)

未经许可,不得转载。 今天我们将分享一个关于在 Bugcrowd 平台的 HubSpot 公共漏洞赏金计划中实现全账户接管的故事。 文章目录 正文SQL 注入主机头污染(Host Header Poisoning)负载均衡器主机头覆盖(Load Balancer Host Header Override)Referer Header 测试ORIGIN Heade…

[笔记] Jenkins 安装与配置全攻略:Ubuntu 从零开始搭建持续集成环境

随着 DevOps 流程的普及,持续集成(CI)和持续交付(CD)已成为现代软件开发中不可或缺的一部分。Jenkins 作为一款开源的自动化服务器,广泛应用于 CI/CD 管道的构建与管理。它不仅支持多种编程语言和工具链&am…

精选2款.NET开源的博客系统

前言 博客系统是一个便于用户创建、管理和分享博客内容的在线平台,今天大姚给大家分享2款.NET开源的博客系统。 StarBlog StarBlog是一个支持Markdown导入的开源博客系统,后端基于最新的.Net6和Asp.Net Core框架,遵循RESTFul接口规范&…

Keepalived 进阶秘籍:全方位配置优化

文章目录 1.sysctl.conf参数优化2. limits参数优化3. global_defs模块区域4.vrrp_script模块区域5.vrrp_instance VI_1实例定义配置模块区域6. virtual_server模块区域7.Keepalived与Heartbeat、Corosync比较 前言: 作为一台Keepalived服务器,有必要对内…

51单片机——中断(重点)

学习51单片机的重点及难点主要有中断、定时器、串口等内容,这部分内容一定要认真掌握,这部分没有学好就不能说学会了51单片机 1、中断系统 1.1 概念 中断是为使单片机具有对外部或内部随机发生的事件实时处理而设置的,中断功能的存在&#…

rabbitmq——岁月云实战笔记

1 rabbitmq设计 生产者并不是直接将消息投递到queue,而是发送给exchange,由exchange根据type的规则来选定投递的queue,这样消息设计在生产者和消费者就实现解耦。 rabbitmq会给没有type预定义一些exchage,而实际我们却应该使用自己定义的。 1.1 用户注册设计 用户在…

基于Python的投资组合收益率与波动率的数据分析

基于Python的投资组合收益率与波动率的数据分析 摘要:該文通过研究马科维茨的投资组合模型,并将投资组合模型应用到包含6只金融股票的金融行业基金中。首先通过开源的财经接口Tushare获取股票原始数据,接着利用数据分析的黄金组合库&#xf…

Linux部署web项目【保姆级别详解,Ubuntu,mysql8.0,tomcat9,jdk8 附有图文】

文章目录 部署项目一.安装jdk1.1 官网下载jdk81.2 上传到Linux1.3 解压1.4 配置环境变量1.5 查看是jdk是否安装成功 二.安装TomCat2.1 官网下载2.2 上传到Linux2.3 解压2.4配置2.5 启动Tomcat2.6 验证是否成功 三.安装mysql四.部署javaweb项目4.1 打包4.2 启动tomcat 部署项目 …

前端基础--网络

http1到http2有哪些新增和区别 HTTP/1.0版本主要增加了 1,增加了HEAD,POST等方法 2,增加了状态码 3,增加了请求头和响应头 4,引入content-type,传输不在仅限于文本 5,在请求中加入了HTTP版本号 HTTP…

Maven 详细配置:Maven 项目 POM 文件解读

Maven 是 Java 开发领域中广泛使用的项目管理和构建工具,通过其核心配置文件——POM(Project Object Model)文件,开发者能够定义项目的基本信息、依赖关系、插件配置以及构建生命周期等关键要素。POM 文件不仅是 Maven 项目的核心…

加速物联网HMI革命,基于TouchGFX的高效GUI显示方案

TouchGFX 是一款针对 STM32 微控制器优化的先进免费图形软件框架。 TouchGFX 利用 STM32 图形功能和架构,通过创建令人惊叹的类似智能手机的图形用户界面,加速了物联网 HMI 革命。 TouchGFX 框架包括 TouchGFX Designer (TouchGFXDesigner)(…

服务器漏洞修复解决方案

漏洞1、远程桌面授权服务启用检测【原理扫描】 Windows Remote Desktop Licensing Service is running: Get Server version: 0x60000604 1、解决方案:建议禁用相关服务避免目标被利用 方法一:使用服务管理器 打开“运行”对话框(WinR&am…

Centos源码安装MariaDB 基于GTID主从部署(一遍过)

MariaDB安装 安装依赖 yum install cmake ncurses ncurses-devel bison 下载源码 // 下载源码 wget https://downloads.mariadb.org/interstitial/mariadb-10.6.20/source/mariadb-10.6.20.tar.gz // 解压源码 tar xzvf mariadb-10.5.9.tar.gz 编译安装 cmake -DCMAKE_INSTA…

基于SpringBoot实现的保障性住房管理系统

🥂(❁◡❁)您的点赞👍➕评论📝➕收藏⭐是作者创作的最大动力🤞 💖📕🎉🔥 支持我:点赞👍收藏⭐️留言📝欢迎留言讨论 🔥🔥&…

Vue进阶(贰幺叁)node 版本切换

文章目录 一、前言1.1 什么是nvm? 二、查看已安装好的 node 版本三、下载 node 版本四、切换 node 版本五、查看在用 node 版本六、拓展阅读 一、前言 项目开发阶段,会涉及多node版本切换应用场景,可应用nvm实现node版本切换。 1.1 什么是nvm? nvm是…

Java-数据结构-链表-高频面试题(1)

在上一篇文章中,我们学习了链表中的"单向链表",但学可不代表就是学会了,能够运用链表的地方比比皆是,解题方法也是层出不穷,今天就让我们巩固一下"单向链表"的知识吧~ 第一题:相交链表…

低空管控技术-无人机云监视技术详解!

一、无人机监听技术的原理 无人机监听技术主要依赖于射频(RF)探测、光学和红外传感器等技术手段。这些技术通过被动监听和监测无人机与飞行员(或控制器)之间的通信链路传输,以确定无人机的位置,甚至在某些…