最近学习Delta3D,  2.4版忙着发布,一直不能成功编译SimCore, 索性静下心来看看源码,官网上竟然提供了几个重要组建的软件设计说明书(SDD),虽说基本都是2005版了,不过我看了后觉得主要构架仍然没有改变, 这几份SDD对于学习Delta3D具有很好的参考价值。

分析源码要从顶层着手,所以在看了GM部分的SDD后,我翻出源码看了下,顺便画几张序列图,算是做个笔记。

分析主要根据TankTargetTutorial(这也是我现在唯一能跑的DEMO),源码版本是09-10-12日的SVN。

Delta3D上层构架主要由 GameManager, GameActor, GameActorProxy, Components这几个组建构成:

 
总体构架图

由上图可看出 GameActor,Components等其他组建都统一由GM(GameManager 下同)管理。
GameManage: 统一管理游戏中的各种资源。完成程序运行时消息处理配送等重要工作,是一个枢纽类

 
GM处理消息

GameActor:主要是游戏中的实际对象,如坦克,飞机等,它需要完成处理与自己相关的消息,更新自我状态等任务。

GameActorProxy: 和GameActor是一对一的关系,既用户每定义一个GameActor,就需要定义一个预期相关的GameActorProxy。其主要任务是提供一个统一的对GameActor的各个属性进行存取操作的接口。接口的统一方便了GM的实现。同时在GM构架中GameActorProxy还起到了消息处理中间层的重要作用。

Components: 是一些游戏的辅助系统,主要功能是在一个较高层次上统一对系统中的信息进行处理(如实现网络通信,用户输入处理等)。

简单的说,使用Delta3D编写游戏,我们只需要实现自己的GameActor,以及Components以完成本应用需要的特定任务,然后按照一定的接口规范将其交给GM管理即可。我们来看一下大致的情况

第一步: 导入Actor

在现有系统下,Delta3D要求用户将自己的GameActor编译成插件形式(Windows下为DLL文件),进而实现能动态载入系统的功能。


导入Actor

   上图显示了Delta3D导入用户定义的GameActor的流程,用户除了要实现自己的GameActor外,还需实现一个ActorPluginRegistry类来作为插件的接口来完成插件注册的工作。很自然,一个ActorPluginRegistry类可以包含多个GameActor对象。

第二步:创建Actor
通过第一步我们将GameActor集以插件的形式导入进来,这时程序就可以创建我们需要的GameActor对象了。

   

创建Actor

   从上图我们可以看出创建GameActor又分两步,第一步是CreateActor, 这一步目的是从刚才导入并注册到系统的插件中找到并创建一个我们想要的GameActor对象。
第二部为AddCreate, 这一步可以理解为将得到的Actor对象加入到游戏场景中。在这一步我们需要完成该GameActor对象的初始化操作, 最重要的任务是对该GameActor需要处理的消息注册到GM中。这一步也是由用户完成的,这样加强了一定的灵活性。

第三步:消息处理流程
有了上面两步,这一步基本上就是Delta3D自己的事情了,我们来看一下Delta3D是如何处理消息的。


消息处理

   上图显示了Delta3D在GM构架下处理消息的流程(仅仅是GameActor接受消息部分)。 一般处理是流程1,这是是GM维护一个消息和GameActor的映射关系表。
然而Delta3D也提供了一定的灵活性,允许GameActor自己同时也维护一个消息和自身的映射表来更灵活的处理消息。如流程2是所示。Delta3D有两套消息处理框架,一套是较低层的以Application类为核心的处理机制,大部分源码包中的Example采用该机制。 一套是较高层的以GameManager类为核心的处理机制,为源码中的Demo所采用。

所有基于Delta3D的程序都要有一套Application处理机制,我们来看下它的实现。


类图

      在上图中需要说明的是, 低层消息基于Base类的消息传送机制(BASE类消息机制可见OSGCHINA社区Delta3D版的一篇文章), BaseABC是一个抽象类,定义了四个纯虚函数。其OnMessage函数执行了将消息分配到这四个函数的工作。 Application类通过实现该四个纯虚函数完成消息处理,同时Application拥有键盘和鼠标的监听器,通过实现并注册相应的回调函数完成键盘,鼠标的输入响应。 GameApplication与Application类的唯一区别就是引入了GM。
程序运行时,GameApplication除了完全照搬执行Application的消息循环外,还通过GM进行高层消息处理循环,也即其同时拥有两套消息机制。
需要注意的是,GM继承自Base,他在构建时向System类进行消息注册,可以直接和系统底层消息队列打交道。
  Delta3D采用OSG进行渲染工作。我们来看一下其低层接口是如何设计的。


接口类图

上图中绿色的类直接提供了对OSG相关对象的封装。 DeltaDrawable类是Delta3D中所有可显示对象的基类(例如Actor, GameActor,Object等),它封装了一个osg:Node。

Delta3D通过Scene类进行场景管理,可以看到Scene和View继承自Base,因而也都拥有了接受,发送低层消息的能力。 因为Delta3D在低层内置了ODE进行物理模拟,Scene类在初始化时向System进行了注册,现在主要是进行接受每帧System发出的消息,调用ODE接口完成物理计算操作。而View类没有通过封装osg::View进行相关视口显示工作,没有重载OnMessage函数,因而不接受消息,主要工作配合Application类处理用户交互的输入。

程序启动时, 有关函数进行关联操作,来看下序列图:


Scene启动配置

从图中可以看到有来该流程有两次绑定工作,第一次将Scene的根节点传给Viewer类,建立了OSG需要的渲染数据链接关系。第二次通过将Viewer对象绑定到Application的CompositeViewer成员,每帧由Application相应的函数执行渲染操作(调用CompositeViewer的Frame函数)。

(转)Delta3D源码分析的更多相关文章

  1. ABP源码分析一:整体项目结构及目录

    ABP是一套非常优秀的web应用程序架构,适合用来搭建集中式架构的web应用程序. 整个Abp的Infrastructure是以Abp这个package为核心模块(core)+15个模块(module ...

  2. HashMap与TreeMap源码分析

    1. 引言     在红黑树--算法导论(15)中学习了红黑树的原理.本来打算自己来试着实现一下,然而在看了JDK(1.8.0)TreeMap的源码后恍然发现原来它就是利用红黑树实现的(很惭愧学了Ja ...

  3. nginx源码分析之网络初始化

    nginx作为一个高性能的HTTP服务器,网络的处理是其核心,了解网络的初始化有助于加深对nginx网络处理的了解,本文主要通过nginx的源代码来分析其网络初始化. 从配置文件中读取初始化信息 与网 ...

  4. zookeeper源码分析之五服务端(集群leader)处理请求流程

    leader的实现类为LeaderZooKeeperServer,它间接继承自标准ZookeeperServer.它规定了请求到达leader时需要经历的路径: PrepRequestProcesso ...

  5. zookeeper源码分析之四服务端(单机)处理请求流程

    上文: zookeeper源码分析之一服务端启动过程 中,我们介绍了zookeeper服务器的启动过程,其中单机是ZookeeperServer启动,集群使用QuorumPeer启动,那么这次我们分析 ...

  6. zookeeper源码分析之三客户端发送请求流程

    znode 可以被监控,包括这个目录节点中存储的数据的修改,子节点目录的变化等,一旦变化可以通知设置监控的客户端,这个功能是zookeeper对于应用最重要的特性,通过这个特性可以实现的功能包括配置的 ...

  7. java使用websocket,并且获取HttpSession,源码分析

    转载请在页首注明作者与出处 http://www.cnblogs.com/zhuxiaojie/p/6238826.html 一:本文使用范围 此文不仅仅局限于spring boot,普通的sprin ...

  8. ABP源码分析二:ABP中配置的注册和初始化

    一般来说,ASP.NET Web应用程序的第一个执行的方法是Global.asax下定义的Start方法.执行这个方法前HttpApplication 实例必须存在,也就是说其构造函数的执行必然是完成 ...

  9. ABP源码分析三:ABP Module

    Abp是一种基于模块化设计的思想构建的.开发人员可以将自定义的功能以模块(module)的形式集成到ABP中.具体的功能都可以设计成一个单独的Module.Abp底层框架提供便捷的方法集成每个Modu ...

随机推荐

  1. 使用Log4j将程序日志实时写入Kafka(转)

    原文链接:使用Log4j将程序日志实时写入Kafka 很多应用程序使用Log4j记录日志,如何使用Kafka实时的收集与存储这些Log4j产生的日志呢?一种方案是使用其他组件(比如Flume,或者自己 ...

  2. Dubbo的一些编码约定和设计原则

    编码约定 代码风格 Dubbo 的源代码和 JavaDoc 遵循以下的规范: Code Conventions for the Java Programming Language How to Wri ...

  3. 同步一个 fork

    fork 了别人的仓库后,原作者又更新了仓库,如何将自己的代码和原仓库保持一致?本文将给你解答. 如何使用搜索引擎 其实这个问题并不难,我又被坑了.百度搜的东西不靠谱啊,以后这种问题一定要用英文在 G ...

  4. NVelocity语法常用指令

    对变量的引用:$ [ ! ][ { ][ a..z, A..Z ][ a..z, A..Z, 0..9, -, _ ][ } ]. 在NVelocity中,对变量的引用都是以$开头加上变量名称.当使用 ...

  5. linux 查看机器内存方法 (free命令)

    工作中遇到了统计机器内存的问题.记录一下. free命令可以查看那机器内存. 如下图单位是M 查看man free可以知道,也可以直接从/proc/meminfo文件中读取.

  6. 排序算法的实现(归并,快排,堆排,希尔排序 O(N*log(N)))

    今天跟着左老师的视频,理解了四种复杂度为 O(N*log(N))的排序算法,以前也理解过过程,今天根据实际的代码,感觉基本的算法还是很简单的,只是自己写的时候可能一些边界条件,循环控制条件把握不好. ...

  7. 移动前端调试工具-Weinre真机调试

    之前做移动前端调试页面的时候就是简单的使用Chrome模拟器调试,能满足基本基本的需求,后来发现了基于Web Inspector(Webkit)的远程调试工具Weinre,可以在PC端直接调试运行在移 ...

  8. Hadoop家族学习路线、实践案例

    作者:Han Hsiao链接:https://www.zhihu.com/question/19795366/answer/24524910来源:知乎著作权归作者所有.商业转载请联系作者获得授权,非商 ...

  9. 为什么static数据成员一定要在类外初始化?(转)

    1.避免重复定义和初始化 <<c++ primer>>说在类外定义和初始化是保证static成员变量只被定义一次的好方法. 但,为什么static const int就可以在类 ...

  10. python3 IEDriver抓取时报数据

    最近做了测试抓取XX时报的数据,由于需要事先登录,并且有验证码,关于验证码解决有两个途径:一是利用打码平台,其原理是把验证码的图片上传发送给打码平台, 然后返回其验证码.二就是自己研究验证码技术问题. ...