喜欢Four这个项目,就赶快在GitHub上Star这个项目吧!
喜欢我的文章,来微博关注我吧:王选易在学C艹
点我下载
项目Logo:
下面是该游戏的项目地址,各位想参考源代码的同学可以到我的GitHub上下载该项目的源码。
项目主页
GitHub地址
bug反馈及建议
我做这个项目的原始目的是实验MVC在游戏中的应用。
Model-View-Controller(MVC)是一种组合设计模式,它体现了一种关注点分离(Separation of concerns,SoC)的思想。MVC主要把逻辑层和表现层进行了解耦,将一个问题划分成了不同的关注点。增强了应用的稳定性,易修改性和易复用性。
MVC经常被使用在Web框架中,包括J2EE,RoR和.Net中都对MVC模型进行了框架层面上的封装,以便程序员可以简单方便地作出结构良好的Web应用。
Cocos2d-x本身并没有提供内置的MVC支持,但是,我们还是可以在游戏中基于MVC架构来设计游戏。在这篇博文中,我将向大家展示一下我是如何使用MVC架构来塔尖Four这个游戏的。
Four这个游戏的创意来自一个叫做走四棋的传统游戏,走四棋规则的详细介绍在这里:走四棋的百度百科。
下面我简单谈一下这个面板游戏(board game)的一些特性
举个例子,下面这幅图即为游戏过程中的一幅图。在下面的游戏过程中,位于(1,0)位置的黑子向左移动到(0,0)的位置后即可吃掉白子。
Cocos2d-x有这样一些主要的类,CCSprite,CCLayer,CCScene,CCNotificationCenter。我们会使用这些类进行游戏中MVC架构的搭建,如果你对这些类的作用不熟,请参考我的这篇博文【Cocos2d-x-基础概念】Director Scene Layer and Sprite。
我们一般的游戏流程是
这个过程看起来十分简单,并且可以十分快速地做出游戏。但是其缺陷就在于在CCLayer中我们做了太多的事情。CCLayer同时承担了逻辑层和表示层的任务。不符合我们上文中提到的关注点分离的原则。如果游戏中有较为复杂的状态转换时就捉襟见肘了。
下面是该游戏项目的目录结构,我们接下来对这几个文件夹进行分别的讲解:
下面是该项目的一个简单的类图
在类图中,虚线代表的是通过消息机制进行沟通,而在Cocos2d-x中,这种沟通是通过CCNotificationCenter来实现的。
Model在游戏中代表的是消息驱动的有限状态机,Model会接受Controller层发送的消息,并根据消息来更改自己的内部数据,然后把内部数据改变这一消息发送给View,通知它更新。
Model在Cocos2d-x对应的是哪个类呢?
很遗憾,但是Cocos2d-x并没有提供状态机的feature,所以我们需要自己实现一个Model类,在Model类中,需要自己实现诸如状态转换和消息处理等功能。例如在我的Model类中,我提供了如下接口。
<!-- lang: cpp -->
class Model : public CCObject {
public:
// 添加一条状态转换,from-起始状态,msg-接收的消息,to-终结状态,在msg发生时会发生状态转换
Model* addTransition(const string& from, const string& msg, const string& to);
// 检查当前状态机能否发生msg对应的状态转换
bool checkMessage(const string& msg);
// 触发msg对应的状态转换
void onMessage(const string& msg);
void onMessage(const string& msg, CCObject* o);
// 等待某个CCAction结束后发送一条消息。
void waitAction(cocos2d::CCNode* node, cocos2d::CCFiniteTimeAction* action, const string& msg);
// 得到当前状态名称
const char* getState();
};
注意以下几点:
CCNotificationCenter
发送一条消息通知游戏中的其他组件更新逻辑。Model不会持有View,所以View都是通过消息来获得Model更新的事件的。
我们在编写游戏时,应该先编写Model,然后通过测试来保证Model的正确性。
之后,我们去写View的时候,实际就是对Model这个状态机发送的各种消息的回调函数的编写了。
这样就讲表示层和逻辑层分离开,可以方便地单独测试每个模块,可维护性大大提高。
在编写Model的时候,我们经常会遇到这样的问题,就是Model中的数据是否应当与View中的数据保持一致。
比如:在View层的棋子的位置信息是否应该和Model层的位置信息保持一致?
其实,真实情况就是,这要看Model层的
计算使用那种数据形式更加方便,比如在Model中,我们会把棋盘转化为一个二维数组,这样在AI运算,逻辑判断时,更加有利,所以棋子的逻辑位置和实际位置必然是不同的。
再比如,很多时候游戏中的物理引擎的计量单位是厘米,米。而不是OpenGL中的坐标。这也是为了逻辑运算的方便。
但是有些数值,我们会让它保持一致,比如人物的属性等等。
View在游戏中代表的是Model消息的接受者,在Cocos2d-x中,View一般是指CCLayer的子节点,即CCSprite,CCLabelAtlas,CCMenuItem,Particle System(粒子系统)等等。
它们会重载onEnter函数,在onEnter中注册自己对某个Message的监听。同时在onExit函数中将所有监听清除。(清除监听很重要,否则会出现很可怕的野指针问题)。
Controller在游戏中负责将用户触摸事件转化为逻辑事件(即我们上文中所说的消息),同时要对用户触摸事件中的信息进行正规化,并且通知变更。
Controller在Cocos2d-x中一般用CCLayer
来实现,因为CCLayer自然地继承了CCTouchDelegate这一接口,本身就可以触摸事件,所以我在这个游戏中所有的XXXController都是继承自CCLayer。
Controller另一个很重要的职责,就是创建View和Model,因为Controller相当于Model和View的一个中间层,所以自然Controller会同时持有View和Model的应用,来方便地传递数据。
Protocol中定义了一些Model,View和Controller中共享的数据,
Message.h
中,就定义了游戏中各种类型的消息,Tag.h
中,就定义了游戏中一些Node的Tag,其实Tag这个定义不是很准确,其实这里的Tag指的是唯一标识CCNode的一个ID。ChessboardProrocol.h
中,定义了游戏中期盘的宽,搞和一些常用的数据结构。