分类:
2009-09-18 20:06:50
在游戏编程精粹四有三篇文章讲到了实体以及实体管理的实现方案,其中一篇文章说到了实体管理系统的四大要素:定义实体怎样沟通的实体消息,实现一实体类代码和数据的实体代码,维护已经注册在案的实体类列表,和用来创建、管理、发送消息的实体管理器。
关于实体消息的内容之前讨论事件机制的时候做过一点说明,其实这也就是按接口调用和按消息驱动的区别,现在mangos的做法是完全的接口调用,所以引擎内部就没有任何的实体消息。实体代码实现和实体管理器是我们重点要讨论的内容。
另有一篇文章也提到了使用类继续的方式实现游戏对象的两大问题,一是它要求系统中的所有对象都必须从一个起点衍生而成,也就是说所有对象类在编译的时候已经确定,这可能是一个不受欢迎的限制,如果开发者决定添加新的对象类,则必须要对基类有所了解,方能支持新类。另一个问题在于所有的对象类都必须实现同样的一些底层函数。
对于第二个问题,可以通过接口继承的方式来避免基类的方法太多。在mangos的实现中就采用了类似的方法,从Object虚基类派生的Unit和WorldObject仍然还是不可实例化的类,这两种对象定义了不同的属性和方法,分来实现不同类型的对象。在游戏内部可以根据对象的实际类型来Object指针向下转型为Unit或WorldObject,以调用需要的接口。方法虽然不够OO,也还能解决问题。但是第一个问题是始终无法避免的。
所以我们便有了通用实体这么一个概念,其主要方法是将原来基类的接口进行分类,分到一个个不同的子类中,然后以对象组合的方式来生成我们所需要的实际游戏对象类型。这个组合的过程可以通过脚本定义的方式,这样便可以在运行时生成为同的对象类型,也就解决了上面提到的第一个问题。
通用实体的实现方法在目前的游戏引擎及开源代码中也可以看到影子。一个是BigWorld,从提供的资料来看,其引擎只提供了一个entity游戏对象,然后由游戏内容实现者通过xml和python脚本来自由定义不同类型的entity类型,每种类型可有不同的property和不同的方法。这样原来由基类定义的接口完全转移到脚本定义,具有非常强的灵活性。
另外还有一个是CEL中的entity实现。按照CEL的描述,entity可以是游戏中的任意对象,包括玩家可交互的对象,如钥匙、武器等,也可以包括不能直接交互的对象,如游戏世界,甚至任务链中的一部分等。entity本身并没有任何特性,具体的功能实现需要靠附加property来完成。简单来说,property才定义了entity可以做什么,至于该怎么做,那又是依靠behavior来定义。所以,最终在CEL中一个游戏对象其实是由entity组合了多个property及多个behavior而生成的。
但是CEL中的property与BigWorld中的property意义不大一样,在CEL中可定义的property其实是引擎内部要先提供的,比如其示例中所举的pcobject.mesh、pcmove.linear、pctools.inventory等,而BigWorld中的property是完全的自由定制。从这个角度来讲,其实可以把CEL中的property看作是游戏的逻辑系统,也就是我们上面所描述的,接口分类后所定义的子类。
由引擎内部提供可选择的property与BigWorld所采用的完全自由定制property其实本质上是相同的。在用BigWorld实现的游戏中,也不可能为每种游戏对象类型都完全从头定义property,基于代码利用的原则,也会先定义一些小类,然后在entity类型定义时以自定义property的方式来包含这些小类。当然,我没有使用过BigWorld,上面的描述也只是基于游戏实现的大原则所做出来的。
描述的依然有些抽象,我们可以用wow及mangos代码来说明一下。mangos中为object定义了一个属性集合,根据对象类型的不同,这个属性集的大小及保存数据也会有差异,另外游戏对象内部含有处理不同游戏逻辑的系统,如RestSystem、FloodFilterSystem、VariousSystem等等,在player.h中以接口组的方式进行定义。
如果要将这种结构改为我们描述的通用entity系统,可以让object只提供property注册和删除的接口,这里的property定义与CEL中的相同,放在mangos中也就是上面说的RestSystem、FloodFilterSystem、VariousSystem这些。然后也通过xml文件的方式定义我们所需要的游戏对象类型,如player,creature,item等,不同的对象类型可以选择加载不同的property,加载的原则是需要哪些功能就加载哪些property。
对象的定义按上面的方法完成后,对象的实现也需要做一些修改。以前客户端的消息是直接交由player来处理,AI也是直接调用creature的接口来完成一些功能,现在通用的entity内部已经没有任何可用的方法,所有的实现都转到了property中,所以需要由各个property实现自己来注册感兴趣的事件及消息,entity实现一个消息的转发,转给对此感兴趣的property来处理。其余的实现就没有什么不同了。
当然,我们再做一点扩展,让property不光由引擎来提供,用脚本本身也能定义property,并且可以通过xml来注册这些property,这样便实现了与BigWorld一样的完全自由特性。这其实也就是将很多用C++实现的功能转移到了python中,这种做法可作为参考,但不一定对所有人合适,至少在我看来,这样的实现也基本只能由程序员来做,所以让程序员选择自己最擅长的语言可能会更易于开发和调试。