浅析cocos2d-x 2.x到3.x事件侦听设计变化原因之一

来源:互联网 发布:淘宝加入购物车的好处 编辑:程序博客网 时间:2024/04/30 19:10

DionysosLai 2015/2/28

对于事件侦听的设计,cocos2d-x从2.x到3.x发生了根本性的变化,一直以来对此,只是单纯的使用考虑如何构建自己的游戏代码,并未对其二者设计孰优孰劣进行探究。只是前段时间在做一个新游戏时,关于2.x的触摸事件发现了一个设计不人性化问题,本想向cocos2dx官网反应,但测试3.x时,并未发现这个问题。对此,本文细述这个问题,分析二者设计的不同,同时希望抛砖引玉。
简单阐述2.x与3.x的事件侦听不同点:
在2.x中,需要注册一个事件侦听。
在3.x中,则需要创建一个侦听器的对象,然后定义回调方法,最后将侦听和事件分发器绑定。
注意,2.x是注册一个事件侦听,在3.x中则是创建一个侦听器对象。这二者有什么不同呢?这里分别在2.x中和3.x写一个测试demo,可以看到2.x中一个严重设计bug问题。

2.x代码如下:

.h    DelegateTest();    virtual ~DelegateTest();    virtual void keyBackClicked();    .cpp    DelegateTest::DelegateTest(){}.cppDelegateTest::~DelegateTest(){    CCLOG("DelegateTest release!!");}bool DelegateTest::init(){    if (!CCLayer::init())    {        return false;    }    this->setKeypadEnabled(true);    CCDirector::sharedDirector()->getTouchDispatcher()->addTargetedDelegate(this, 0 ,true);    return true;}void DelegateTest::keyBackClicked(){    CCScene* sceneGame = DelegateTest::scene();    CCDirector::sharedDirector()->replaceScene(sceneGame);    CCLog("keyBackClicked!!!");}

这里,是一个简单的触摸侦听事件事件demo,在init()中添加注册触摸事件,同时使能返回键功能,同时按下返回键时,切换界面。
先理论分析调试信息,应该是如下所示:
按下返回键:
keyBackClicked!!!
DelegateTest release!!
再次按下返回键:
keyBackClicked!!!
DelegateTest release!!
……
事件呢,调试信息如下:
按下返回键:
keyBackClicked!!!
再次按下返回键:
keyBackClicked!!!
退出游戏:
DelegateTest release!!
DelegateTest release!!
看到区别了吗!出现这种现象的结果什么呢。就是当前场景并未移除掉,同样能接收触摸,当不止一个场景出现时,其逻辑就会出现紊乱!!!
现在代码移植到3.x中。

3.x代码如下:

.h    DelegateTest();    virtual ~DelegateTest();    virtual void onKeyReleased(cocos2d::EventKeyboard::KeyCode keyCode, cocos2d::Event* event) override;    cocos2d::EventListenerKeyboard* m_keyboardListener;     // 键盘listener    cocos2d::EventListenerTouchOneByOne* m_touchListener    ;   // 触摸listener.cppDelegateTest::DelegateTest(){}DelegateTest::~DelegateTest(){    log("DelegateTest release!!");}bool DelegateTest::init(){    if (!CCLayer::init())    {        return false;    }    m_keyboardListener = EventListenerKeyboard::create();    m_keyboardListener->retain();    m_keyboardListener->onKeyReleased = CC_CALLBACK_2(DelegateTest::onKeyReleased, this);    getEventDispatcher()->addEventListenerWithSceneGraphPriority(m_keyboardListener, this);    //setKeypadEnabled(true); ----不再适用    m_touchListener = EventListenerTouchOneByOne::create();    m_touchListener->retain();    m_touchListener->onTouchBegan = CC_CALLBACK_2(DelegateTest::onTouchBegan, this);    m_touchListener->onTouchBegan = [&](cocos2d::Touch* touch, cocos2d::Event* event){        return true;    };    getEventDispatcher()->addEventListenerWithSceneGraphPriority(m_touchListener, this);    return true;}void DelegateTest::onKeyReleased( cocos2d::EventKeyboard::KeyCode keyCode, cocos2d::Event* event ){    Scene* sceneLayer = DelegateTest::scene();    Director::getInstance()->replaceScene(sceneLayer);    log("onKeyReleased!!!");}

功能与前面描述的一模一样。调试信息如下所示:
onKeyReleased!!!
DelegateTest release!!
onKeyReleased!!!
DelegateTest release!!
与我们要的结果一致。为何会出现这两种不同的情况呢。这是由于2.x中,是注册侦听事件,因此我们必须对应的要注销侦听事件。而3.x是创建侦听器对象,这样析构时,会自动移除对象,从设计上来说,3.x比2.x更加的容易,且不容易出错。
那么对于2.x要如何更改,使之出现的结果跟我们理论上一致呢?这里更改函数keyBackClicked()代码即可:

void DelegateTest::keyBackClicked(){    CCScene* sceneGame = DelegateTest::scene();    CCDirector::sharedDirector()->replaceScene(sceneGame);    CCDirector::sharedDirector()->getTouchDispatcher()->removeDelegate(this);    CCLog("keyBackClicked!!!");}

当然,理论上3.x退出场景时,也要移除侦听器对象,函数是release(),这里省咯了而已。也许大家会说,不就是在2.x中添加一句话而已,而且忘记注销,本身就是自己的问题。这里首先要说一点,就是当layer比较多时,注销触摸并不是一件简单的事情,同时,退出场景可能在一个子类中调用;也可能在另一个layer中,要求关闭当前场景中所有layer,比如暂停界面。另一个面,当使用到popScene和pushScene时,这更加是一个灾难!!!对于一个游戏引擎,良好的设计应该能主动帮忙开发者避免以坑。这也是为什么在cocos2d-x中渲染时,为何不专门开一个线程,实现逻辑与展示分开。因为这样太难了,会导致大量细节问题,无形中加大开发者难度,当然,随着以后引擎开发,相信会解决这个问题。详细可以参考王哲回答:http://www.zhihu.com/question/27612727/answer/38078748。
对于以上代码细节,可以访问笔者githup:https://github.com/DionysosLai/CocoSamples 中的“Delegate Test”内容。2015年第一篇文章,新年新兆头,祝大家新年更棒!!!—ps:第一次使用markdown编辑器,感觉萌萌哒!!!

1 0
原创粉丝点击