




已阅读5页,还剩10页未读, 继续免费阅读
版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领
文档简介
游戏用到的模式使用设计模式来提高程序库的重复利用性是大型程序项目开发必须的。但是在“四人帮”的设计模式概述中提到了23种标准设计模式,不但难以记住,而且有些设计模式更多的适用于应用程序开发,对游戏项目引擎设计并没有很多的利用价值。根据经验,精挑细选后,笃志在这里记录一些自认为有利用价值的设计模式,以便之后自己设计时使用。一:观察者Observer观察者的设计意图和作用是:它将对象与对象之间创建一种依赖关系,当其中一个对象发生变化时,它会将这个变化通知给与其创建关系的对象中,实现自动化的通知更新。游戏中观察者的适用环境有:1:UI控件管理类。当我们的GUI控件都使用观察者模式后,那么用户的任何界面相关操作和改变都将会通知其关联对象我们的UI事件机。2:动画管理器。很多时候我们在播放一个动画桢的时候,对其Frame有很大兴趣,此时我们设置一个FrameLister对象对其进行监视,获得我们关心的事件进行处理是必须的。观察者伪代码:/ 被观察对象目标类Class Subject/ 对本目标绑定一个观察者 Attach( Observer );/ 解除一个观察者的绑定 DeleteAttach( Observer );/ 本目标发生改变了,通知所有的观察者,但没有传递改动了什么Notity()For ( 遍历整个ObserverList ) pObserver -Update(); / 对观察者暴露的接口,让观察者可获得本类有什么变动GetState();/ 观察者/监听者类Class Observer/ 暴露给对象目标类的函数,当监听的对象发生了变动,则它会调用本函数通知观察者Void Update ()pSubject -GetState(); / 获取监听对象发生了什么变化TODO:DisposeFun(); / 根据状态不同,给予不同的处理/非程序语言描述:A是B的好朋友,对B的行为非常关心。B要出门,此时A给了B一个警报器,告诉B说:“如果你有事,立刻按这个警报器告诉我。”。结果B在外面遇上了麻烦,按下警报器(Update(),B就知道A出了事,于是就调查一下B到底遇到了什么麻烦(GetState(),当知道B原来是因为被人打了,于是立刻进行处理DisposeFun(),派了一群手下帮B打架。当然关心A的人可以不止一个,C,D可能也对A很关心,于是A这里保存一个所有关心它的人的链表,当遇到麻烦的时候,轮流给每个人一份通知。 二:单件模式Singleton 单件模式的设计意图和作用是:保证一个类仅有一个实例,并且,仅提供一个访问它的全局访问点。游戏中适用于单件模式的有:1:所有的Manger.在大部分的流行引擎中都存在着它的影子,例如SoundManager, ParticeManager等。2:大部分的工厂基类。这一点在大部分引擎中还是见不到的,实际上,我们的父类工厂采用唯一实例的话,我们子类进行扩展时也会有很大方便。单件模式伪代码:/Class SingletonStatic MySingleton; / 单件对象,全局唯一的。Static Instance() return MySingleton; / 对外暴露接口/三:迭代器Iterator迭代器设计意图和作用是:提供一个方法,对一个组合聚合对象内各个元素进行访问,同时又不暴露该对象类的内部表示。游戏中适用于迭代器模式的有:因为STL的流行,这个设计已经广为人知了,我们对任何形式的资源通一管理时,不免会将其聚合起来,或者List,或者Vector,我们都需要一个对其进行访问的工具,迭代器无疑是一个利器。迭代器伪代码:/ 迭代器基类Class IteratorVirtual First();Virtual Next();Virtual End();Virtual CurrentItem(); / 返回当前Item信息/ 聚合体的基类Class ItemAggregateVirtual CreateIterator(); / 创建访问自身的一个迭代器/ 实例化的项目聚合体Class InstanceItemAggregate : public ItemAggregateCreateIterator() return new InstanceIterator(this); / 四:访问者模式Visitor:访问者设计意图和作用是:当我们希望对一个结构对象添加一个功能时,我们能够在不影响结构的前提下,定义一个新的对其元素的操作。(实际上,我们只是把对该元素的操作分割给每个元素自身类中实现了而已)游戏中适用于访问者模式的有:任何一个比较静态的复杂结构类中都适合采用一份访问者。这里的“比较静态的复杂结构类”意思是,该结构类中元素繁多且种类复杂,且对应的操作较多,但类很少进行变化,我们就能够将,对这个结构类元素的操作独立出来,避免污染这些元素对象。1:例如场景管理器中管理的场景节点,是非常繁多的,而且种类不一,例如有Ogre中的Root, Irrchit中就把摄象机,灯光,Mesh,公告版,声音都做为一种场景节点,每个节点类型是不同的,虽然大家都有共通的Paint(),Hide()等方法,但方法的实现形式是不同的,当我们外界调用时需要统一接口,那么我们很可能需要需要这样的代码Hide( Object ) if (Object = Mesh) HideMesh(); if (Object = Light) HideLight(); 此时若我们需要增加一个Object新的类型对象,我们就不得不对该函数进行修正。而我们可以这样做,让Mesh,Light他们都继承于Object,他们都实现一个函数Hide(),那么就变成Mesh:Hide( Visitor ) Visitor.Hide (Mesh); Light:Hide(Visitor ) Visitor.Hide (Light); 我们在调用时只需要Object.Hide(Visitor) return Visitor.Hide(Object); 这样做的好处,我们免去了对重要函数的修正,Object.Hide(Visitor)函数我们可以永久不变,但是坏处也是很明显的,因为将方法从对象集合结构中抽离出来,就意味着我们每增加一个元素,它必须继承于一个抽象的被访问者类,实现其全部函数,这个工作量很大。所以,访问者是仅适合于一个装载不同对象的大容器,但同时又要求这个容器的元素节点不应当有大的变动时才使用。另外,废话一句,访问者破坏了OO思想的。访问者伪代码:/ 访问者基类Class VisitorVirtual VisitElement( A ) ; / 访问的每个对象都要写这样一个方法Virtual VisitElement( B ) ;/ 访问者实例AClass VisitorAVisitElement( A ) ; / 实际的处理函数VisitElement( B ) ; / 实际的处理函数/ 访问者实例BClass VisitorBVisitElement( A ) ; / 实际的处理函数VisitElement( B ) ; / 实际的处理函数/ 被访问者基类Class ElementVirtual Accept( Visitor ); / 接受访问者/ 被访问者实例AClass ElementAAccecpt( Visitor v ) v- VisitElement(this); ; / 调用注册到访问者中的处理函数/ 被访问者实例BClass ElementBAccecpt( Visitor v ) v- VisitElement(this); ; / 调用注册到访问者中的处理函数/ 五:外观模式Facade外观模式的设计意图和作用是:将用户接触的表层和内部子集的实现分离开发。实际上,这个模式是个纸老虎,之后我们看伪代码立刻就会发现,这个模式实在用的太频繁了。游戏中需要使用外观模式的地方是:这个非常多了,举几个比较重要的。1:实现平台无关性。跨平台跨库的函数调用。2:同一个接口去读取不同的资源。3:硬件自动识别处理系统。外观模式伪代码/ 用户使用的接口类Class Interface/ 暴露出来的函数接口函数,有且仅有一个,但内部实现是调用了两个类Void InterfaceFun()/ 根据某种条件,底层自主的选择使用A或B的方法。用户无须关心底层实现If ( XXX )ActualA-Fun();ElseActualB-Fun();/ 实际的实现,不暴露给用户知道Class ActualAVoid Fun();/ 实际的实现,不暴露给用户知道Class ActualBVoid Fun();怎么样,纸老虎吧,看起来很高深摸测的命名而已。/ 六:抽象工厂模式AbstractFactory抽象工厂的设计意图和作用是:封装出一个接口,这个接口负责创建一系列互相关联的对象,但用户在使用接口时不需要指定对象所在的具体的类。从中文命名也很容易明白它是进行批量生产的一个生产工厂的作用。游戏中使用抽象工厂的地方有:基本上任何有批量的同类形式的子件地方就会有工厂的存在。(补充一句:下面代码中的ConcreteFactory1实例工厂就是工厂,而抽象工厂仅仅是工厂的一个抽象层而已。)1:例如,在音频方面,一个音频的抽象工厂派生出不同的工厂,有音乐工厂,音效工厂。音效工厂中又有一个创建3D音效节点的方法,一个创建普通音效节点的方法。最终用户只需要SoundFactory-Create3DNode( pFileName );就可以创建一个节点了。2:场景对象。3:渲染对象。4:等等工厂与单件,管理器Manager关系一定是非常紧密的。抽象工厂伪代码:/class AbstractProductA ; / 抽象的产品A基类class AbstractProductB ; /抽象的产品B基类/ 抽象工厂基类class AbstractFactory public:virtual AbstractProductA* CreateProductA() = 0 ;/ 创建ProductA virtual AbstractProductB* CreateProductB() = 0 ;/ 创建ProductB ;class ProductA1 : public AbstractProductA ; / 产品A的实例1 class ProductA2 : public AbstractProductA ; / 产品A的实例2class ProductB1 : public AbstractProductB ; / 产品B的实例1 class ProductB2 : public AbstractProductB ; / 产品B的实例2/ 实例工厂1class ConcreteFactory1 : public AbstractFactory virtual AbstractProductA* CreateProductA() return new ProductA1() ; virtual AbstractProductB* CreateProductB() return new ProductB1() ; static ConcreteFactory1* Instance() / 实例工厂尽量使用单件模式 ;/ 实例工厂2class ConcreteFactory2 : public AbstractFactory virtual AbstractProductA* CreateProductA() return new ProductA2() ; virtual AbstractProductB* CreateProductB() return new ProductB2() ; static ConcreteFactory2* Instance() / 实例工厂尽量使用单件模式 ;/客户端代码:Void main() AbstractFactory *pFactory1 = ConcreteFactory1:Instance() ;AbstractProductA *pProductA1
温馨提示
- 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
- 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
- 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
- 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
- 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
- 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
- 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。
最新文档
- 中级财务会计(河南财经政法大学)知到智慧树答案
- 死因监测培训试题及答案
- 2025大连安居客平台全程监管下的二手房买卖合同
- 2025年度绿色金融垫资撤押贷款合同及碳排放权抵押担保协议
- 2025版蔬菜种植与农产品电商平台合作合同
- 2025年新型农业灌溉系统安装与运营管理合同
- 2025年水利工程桩基施工与生态修复合同
- 2025版跨境电商合作万能合同范本
- 数据驱动的实时监控与异常检测-洞察及研究
- 2025二手公寓房买卖及贷款担保与房屋租赁服务合同
- 2025年学历类自考专业(学前教育)学前儿童发展-学前教育原理参考题库含答案解析(5套)
- 日本设备销售合同范本
- 2025年芜湖市鸠江区医院招聘16名工作人员笔试参考题库附答案解析
- T-CBDA 86-2025 建筑幕墙、采光顶及金属屋面工程质量验收标准
- 厨房消防安全培训
- 小陈 税务风险应对常见指标与答复思路
- 2025年《中华人民共和国档案法》知识培训试题及答案
- 2025至2030年中国建筑膜行业市场调查研究及发展趋势预测报告
- 变电站新员工培训课件
- 《海上风电场工程测量规程》(NB-T 10104-2018)
- 2021年成都中医药大学辅导员招聘笔试试题及答案解析
评论
0/150
提交评论