Cocos2dx的计时器schedule,在《【Cocos2dx】连续滚动的场景》(点击打开链接)中实现即时更新事件的时候已经提到过,但是当时只是用到this->schedule(),这个无参数的计时器,仅仅是实现不停执行函数的功能,如果要求定间隔执行一段函数,延时执行一段代码,则需要对this->schedule()进行更加详细的运用。
下面用一个小例子说明Cocos2dx的计时器如何使用。
如下图:
给Helloworld上一个每0.05s执行一次的定时器,这比较等同于this->schedule(),在每0.05s,按钮精灵则会向右移动2像素,虽然这更适合可以用《【Cocos2dx】基本动作、动作序列与动作合并》(点击打开链接)去做,但是这是为了说明Cocos2dx中计时器的使用,因此这样搞。
如果这个按钮越过屏幕的1/4位置,则在右上角添加一个数字,开启一个每次1s倒数1的计时器,同时开启一个3s后终止所有计时器的定时器。
Helloworld.h的声明如下:
#include "cocos2d.h" USING_NS_CC;//用到了CCxx,比如CCNode class HelloWorld : public cocos2d::CCLayer { public: virtual bool init(); static cocos2d::CCScene* scene(); CREATE_FUNC(HelloWorld); private: CCSprite* sprite;//按钮精灵 CCLabelTTF *label;//右上角的文字 int counter;//3、2、1、0倒数 bool count_backwards_timer_open;//每1s倒数一次的计时器 是否打开的标识 bool delay_action_open;//3s后停止所有计时器 的 计时器 是否打开的标识 void count_backwards_timer(float delta);//每1s倒数一次的计时器 具体的方法 void timer(float delta);//每0.05s移动按钮精灵 具体的方法 void delay_action(float delta);//3s后停止所有计时器 具体的方法 };
从头文件的声明就已经可以看到,有两个计时器是否打开的标识。这是防止这两个计时器重复打开。
具体见Helloworld.cpp的代码:
#include "HelloWorldScene.h" CCScene* HelloWorld::scene() { // 'scene' is an autorelease object CCScene *scene = CCScene::create(); // 'layer' is an autorelease object HelloWorld *layer = HelloWorld::create(); // add layer as a child to scene scene->addChild(layer); // return the scene return scene; } //HelloWorld场景初始化之时 bool HelloWorld::init() { CCSize visibleSize=CCDirector::sharedDirector()->getVisibleSize();//获取屏幕的尺寸、位置信息等 /*按钮精灵的声明*/ sprite=CCSprite::create("CloseSelected.png"); sprite->setPosition(ccp(0,visibleSize.height/2)); this->addChild(sprite); this->schedule(schedule_selector(HelloWorld::timer),0.05f);//开启一个计时器,此计时器每0.05s执行一次 /*初始化标识*/ count_backwards_timer_open=false; bool delay_action_open=false; return true; } void HelloWorld::timer(float delta){//这段代码每0.05s执行一次 CCSize visibleSize=CCDirector::sharedDirector()->getVisibleSize();//获取屏幕的尺寸、位置信息等 sprite->setPositionX(sprite->getPositionX()+2);//按钮精灵向右移动2像素 if(sprite->getPositionX()>visibleSize.width/4){//如果按钮精灵超过屏幕的1/4 if(!count_backwards_timer_open){//且倒数的计时器没打开 /*在右上角初始化一个标签文本*/ count_backwards_timer_open=true;//改变标识,关闭这个入口 label=CCLabelTTF::create("3","arial",36); counter=3; label->setAnchorPoint(ccp(1,1)); label->setPosition(ccp(visibleSize.width,visibleSize.height)); this->addChild(label); this->schedule(schedule_selector(HelloWorld::count_backwards_timer),1.0f);//开启一个每1s执行一次的计时器 } if(delay_action_open){//如果3s后停止所有计时器 没打开 //则开启一个3s后停止所有计时器 的定时器 delay_action_open=true; this->scheduleOnce(schedule_selector(HelloWorld::delay_action),3.0f);//注意!这里的方法从this->schedule变成了this->scheduleOnce } } } void HelloWorld::count_backwards_timer(float delta){//这段代码每1s会被执行一次 counter--; label->setString(CCString::createWithFormat("%d",counter)->getCString());//CCString::createWithFormat("%d",counter)->getCString()是Cocos2dx自带的整形等转字符串、连接字符串的方法,由于C++的整形等转字符串、处理字符串起来非常复杂,这能这样搞 } void HelloWorld::delay_action(float delta){//这段代码会在开启计时器之后延迟3秒后才执行,仅执行一次 label->setString(CCString::createWithFormat("%d",0)->getCString()); this->unscheduleAllSelectors();//停止此时此刻的所有计时器,所有计时器都不会被执行 }
从上述代码大家应该可以为何要单独设立两个嵌套在计时器中的 计时器 是否打开的标识,
如果没有涉及标识是否打开的条件结构,那么两个嵌套在计时器中的 计时器 会被不停地打开,导致程序错乱,崩溃。
之所以会不停地打开,是因为this->schedule(...)这个打开计时器的函数由于本身 处于 计时器中 ,计时器的所有代码,每 x 秒则会被执行一次。
因此计时器的嵌套是需要注意。
而schedule_selector是能够指明所执行的代码,后面的浮点型除了指明执行时间间隔,还会被传递到相应计时器的实现函数的参数float delta中,虽然这个参数没有什么用。
this->unscheduleAllSelectors();是停止所有计时器,想单独停止某一计时器,用this->unschedule(scheducle_selector(xx函数));
其实计时器就是涉及伟大的操作系统的线程问题。