积累提供所有操作(的实现)来定义子类的行为
用一个最简单的例子来讲解这个模式
玩家操纵的英雄也就是这个游戏的主角会有许多技能,我们想定义许多不同的技能,来让玩家使用。

先我们定义一个skillBase类作为基类,我们所有技能的动作都在这里实现。我们可以从这些基本元动作中组合出各种各样的技能,甚至成百上千种,可以
设计一个doc文档来设计各种技能的操作,及操作顺序。这就是之所以为什么叫子类沙盒的原因,把实现技能的方法作为沙盒,向这个沙盒里加入各种各样的元动
作来组成各种各样的技能。
以传说系列的凤凰天驱为例,如下图(源自世界传说-换装迷宫2),这个技能分为,后移、跳跃、播放动画、前冲、粒子效果、播放动画等等元动作组成的

如果我们不使用子类沙盒模式,而一个一个写技能的话会有如下缺点:
1.    会产生大量重复代码,造成代码冗余,因为每个技能都有重复的地方,比如说播放声音,播放动画等。使用这种模式之后各个操作方法就像一个一个的组件一样,随意使用,不会有重复。
2.

每一个技能类都会与游戏系统和游戏引擎耦合,比如声音、动画播放,而如果把操作实现都写在基类里让子类组合的话就只有基类与之耦合而已。这个原因这种模
式带来的好处是,如果有某个操作增删改的话,就不用每个技能类都增删改一遍,而只改基类就好,方便简单,简约。而且重用性相当好,这些代码能用在大量游戏
上(比如说传说系列每个系列的传承->魔神剑)
在基类skillBase中,我们需要实现一些技能的元动作,比如move移动,jump跳,playAnimation播放动画,playSound播放声音等等,我们把他们组合在一起实现各种技能,这些方法是受保护的。
然后我们需要一个virtual 方法action是最终的技能在子类中定义实现,这就是为什么上面的原方动作是受保护的了,因为我们不需要调用这些元动作,只需要他们组合成的技能action()方法就好,所以只让子类获取元动作方法用protected标记。
于是乎我们做一些受保护的方法,在子类中拼装实现一个组合在一起的整体。
我们要做这些技能类
1.    建立一个继承于基类skillBase的技能类起名为skill1
2.    重写skillBase的action方法
3. 把元动作在action中实现

类(skillBase)中提供了一个抽象的沙盒方法(action)和一些标记为protected的元动作操作(move、jump、
playSound等),这个类派生了一些沙盒子类,每个沙盒子类(skill1、skill2。。。)都实现了沙盒方法(action),沙盒方法包含
着各种元动作(move、jump、playSound等)。
当符合以下条件时可以使用子类沙盒模式
1.    一个基类和大量派生子类
2.    基类可以提供所有派生子类需要的操作
3.    子类之间的操作、方法有重复
4.    想要减少子类和游戏系统、游戏引擎等的耦合

注意:因为此时子类与基类密切相关,所以可能会产生brittle base class问题,也就是你看似安全的修改了基类,但是子类却可能因此发生问题。

代码实现

代码如下:
在skillBase中简单实现几个元动作和抽象的沙盒方法:

   protected void act(int actID)
{
hero.Act(actID);//播放攻击动画
}
protected void playSound(int soundID)
{
audioSource.PlayOneShot(audio[soundID]);
} protected void move(Vector3 dir, float moveSpeed, bool isRun)
{
moveFunc.move(dir, moveSpeed, isRun);
}
protected void jump()
{
moveFunc.jump();
}
protected void particalEffect()
{
。。粒子效果。。
} virtual public void action()//沙盒方法
{ }

再看skill1实现的沙盒方法,比如说我们想实现一个跳斩,就把这些元动作组合在一起

    override public void action()
{
playSound();
act();
jump();
particalEffect();
}

很简单对吧
我们再来看看基类初始化的方法
基类的初始化
方法一:构造函数
通过构造函数传参来赋值,但是这种有参构造函数必须要在每个沙盒子类中base一下,所以如果base类的构造函数这些参数有个增删改,子类也得跟着全部增删改。

    public SkillBase(AudioSource _audioSource, AudioClip[] _audio, Move1 _moveFunc, Hero2 _hero, GameObject _heroObject)
{
this.audio = _audio;
this.audioSource = _audioSource;
this.moveFunc = _moveFunc;
this.hero = _hero;
this.heroObject = _heroObject;
}

方法二:初始化函数init
不过记得要在一开始调用,否则小心游戏崩溃

    public void init(AudioSource _audioSource, AudioClip[] _audio, Move1 _moveFunc, Hero2 _hero, GameObject _heroObject)
{
this.audio = _audio;
this.audioSource = _audioSource;
this.moveFunc = _moveFunc;
this.hero = _hero;
this.heroObject = _heroObject;
}

方法三:静态初始化方法init和类变量(静态变量)

这样所有沙盒子类也就是所有skill都共用一种变量了,有好处也有坏处

   static private AudioSource audioSource;
static private AudioClip[] audio;
static private Move1 moveFunc;
static private Hero2 hero;
static protected GameObject heroObject; static public void init(AudioSource _audioSource, AudioClip[] _audio, Move1 _moveFunc, Hero2 _hero, GameObject _heroObject)
{
audio = _audio;
audioSource = _audioSource;
moveFunc = _moveFunc;
hero = _hero;
heroObject = _heroObject;
}

总结

小模式,大用处。很简单的一个设计模式,却很有用处,成功的减少了沙盒子类与其他类之间的耦合,也增加了重用性,更重要的是这种组合的方式增加了多样性,这正是游戏所需要的。

全部代码已上传至GitHub

博主近期渲染:最近用unity5弄的一些渲染

---- by wolf96 

游戏开发设计模式之子类沙盒模式(unity3d 示例实现)的更多相关文章

  1. 游戏开发设计模式之状态模式 & 有限状态机 & c#委托事件(unity3d 示例实现)

    命令模式:游戏开发设计模式之命令模式(unity3d 示例实现) 对象池模式:游戏开发设计模式之对象池模式(unity3d 示例实现) 原型模式:游戏开发设计模式之原型模式 & unity3d ...

  2. 游戏开发设计模式之原型模式 & unity3d JSON的使用(unity3d 示例实现)

    命令模式:游戏开发设计模式之命令模式(unity3d 示例实现) 对象池模式:游戏开发设计模式之对象池模式(unity3d 示例实现) 实现原型模式 原型模式带来的好处就是,想要构建生成任意独特对象的 ...

  3. 游戏开发设计模式之对象池模式(unity3d 示例实现)

    前篇:游戏开发设计模式之命令模式(unity3d 示例实现) 博主才学尚浅,难免会有错误,尤其是设计模式这种极富禅意且需要大量经验的东西,如果哪里书写错误或有遗漏,还请各位前辈指正. 原理:从一个固定 ...

  4. APNS IOS 消息推送沙盒模式和发布模式

    在做.NET向IOS设备的App进行消息推送时候,采用的是PushSharp开源类库进行消息的推送,而在开发过程中,采用的是测试版本的app,使用的是测试的p12证书采用的是ApnsConfigura ...

  5. 游戏开发设计模式之命令模式(unity3d 示例实现)

    博主才学尚浅,难免会有错误,尤其是设计模式这种极富禅意且需要大量经验的东西,如果哪里书写错误或有遗漏,还请各位前辈指正. 打 算写设计模式的目的就是,首先自己可以理清思路,还有就是国内的设计模式资料很 ...

  6. iOS 开发查看应用的沙盒文件

    在iOS开发中,常常需要将一些信息保存到本地,比如说用户的一些搜索历史等.那么,如何查看所保存的文件呢? 这里介绍两种途径来查看应用的沙盒文件. 方法一:通过Xcode来查看,步骤如下: (1): X ...

  7. JavaScript 沙盒模式

    微前端已经成为前端领域比较火爆的话题,在技术方面,微前端有一个始终绕不过去的话题就是前端沙箱 什么是沙箱 Sandboxie(又叫沙箱.沙盘)即是一个虚拟系统程序,允许你在沙盘环境中运行浏览器或其他程 ...

  8. iOS开发 - 获取真机沙盒数据

    今天要获取之前真机測试时写入沙盒的数据, 本来以为挺麻烦的. 后来捣腾了一下, 才知道原来这么简单... 以下直接看详细步骤. 前提: 真机已经通过USB和你的电脑连接上了! 1.进入Organize ...

  9. 开发设计模式(九)门面模式(Facade Pattern)

    什么是门面模式? 门面模式要求一个子系统的外部与其内部的通信必须通过一个统一的门面(Facade)对象进行.门面模式提供一个高层次的接口,使得子系统更易于使用. 大家都写过纸质的信件吧,比如给女朋友写 ...

随机推荐

  1. U3D C#脚本的生命周期

    MonoBehaviour是每个脚本的基类. 每个Javascript脚本自动继承MonoBehaviour,使用C#或Boo时,需要显式继承MonoBehaviour. 一开始实例化,直到结束实例被 ...

  2. c语言学习之基础知识点介绍(十):数组

    本节主要介绍数组. 一.数组 /* 数组:一个变量可以存n个变量. 语法:类型 数组名[长度(正整数)]; 例如:int score[5];//定义了一个int类型的数组,长度为5,可以保存5个数据. ...

  3. jquery table的隔行变色 鼠标事件

    一.鼠标事件 mouseover(function(){}); 鼠标移动到目标事件 mouseout(function(){}); 鼠标离开目标的事件 二.具体应用代码 <body> &l ...

  4. java中 引用类型 和 基本类型 有何区别?

    栈与堆都是Java用来在Ram中存放数据的地方.与C++不同,Java自动管理栈和堆,程序员不能直接地设置栈或堆. Java的堆是一个运行时数据区,类的(对象从中分配空间.这些对象通过new.newa ...

  5. Lucene分页-----SearcherAfter

    /** * 分页,SearcherAfter * @param query * @param pageIndex * @param pageSize */ public void searchPage ...

  6. MATLAB中的函数的归总

    字符串操作函数 1.        函数eval可以用来执行用字符串表示的表达式 2.        函数deblank可以去掉字符串末尾的所有空格 3.        函数findstr可以用来在长 ...

  7. mysql 刘道成视频教程 第4-8课 --- 数据类型

    数据类型大纲图: 注:在mysql中,输入时,除了数值型,不要加单引号,其他的都要加上单引号,养成一种好习惯. 一.数值型: 整数型: 1)从数学上来讨论tinyint 1. 占据空间 2.存储范围 ...

  8. OSPF + LVS ,突破LVS瓶颈 (转)

    突破LVS瓶颈,LVS Cluster部署(OSPF + LVS) 前言 架构简图 架构优势 部署方法 1.硬件资源准备 2.三层设备OSPF配置 3.LVS调度机的OSPF配置 a.安装软路由软件q ...

  9. 安装sql server 2008,提示要删除SQL Server 2005 Express 工具 怎么解决?

    x86 修改注册表:HKLM\Software\Microsoft\Microsoft SQL Server\90\Tools\ShellSEM,把 ShellSEM重命名即可. x64       ...

  10. js简版定时器

    <html> <head> <meta http-equiv="Content-Type" content="text/html; char ...