上篇文章分析到了定时器的定义。这篇的重点就是定时器是怎样执行起来的。

1.从main中寻找定时器的回调

讲定时器的执行,就不得不触及到cocos2dx的main函数了,由于定时器是主线程上执行的。并非单独线程的。所以它的调用必定会在main函数中,每帧调用。

下面代码就是win32平台下的main函数

int APIENTRY _tWinMain(HINSTANCE hInstance,
HINSTANCE hPrevInstance,
LPTSTR lpCmdLine,
int nCmdShow)
{
UNREFERENCED_PARAMETER(hPrevInstance);
UNREFERENCED_PARAMETER(lpCmdLine); // create the application instance
AppDelegate app;
return Application::getInstance()->run();
}

直接调用了run函数,直接进入到run中

int Application::run()
{
while(!glview->windowShouldClose())
{
QueryPerformanceCounter(&nNow);
if (nNow.QuadPart - nLast.QuadPart > _animationInterval.QuadPart)
{
nLast.QuadPart = nNow.QuadPart - (nNow.QuadPart % _animationInterval.QuadPart); director->mainLoop(); //看这里
glview->pollEvents();
}
else
{
Sleep(1);
}
}
return 0;
}

run函数中其它不相关的调用我已经去掉了,能够看到mainLoop函数才是真正的主循环

void DisplayLinkDirector::mainLoop()
{
//做其它不相关的事情
if (! _invalid)
{
drawScene();
}
}

看到这里实际上也是调用drawScene函数

void Director::drawScene()
{
if (! _paused)
{
_scheduler->update(_deltaTime);
_eventDispatcher->dispatchEvent(_eventAfterUpdate);
}
//之后才进行绘制
}

drawScene要做的事情非常多,我将绘制部分都去掉了。值得注意的是 绘制场景会在定时器之后才运行。

这里能够看到,实际上运行的就是定时器的update函数。那么这个update函数中到底运行了什么东西呢?

2.定时器的update函数

首先来看看Update的代码

// main loop
void Scheduler::update(float dt)
{
_updateHashLocked = true; if (_timeScale != 1.0f)
{
dt *= _timeScale;
} //
// 定时器回调
// tListEntry *entry, *tmp; // Update定时器中优先级小于0的队列先运行
DL_FOREACH_SAFE(_updatesNegList, entry, tmp)
{
if ((! entry->paused) && (! entry->markedForDeletion))
{
entry->callback(dt);
}
} // 接下来是优先级等于0的
DL_FOREACH_SAFE(_updates0List, entry, tmp)
{
if ((! entry->paused) && (! entry->markedForDeletion))
{
entry->callback(dt);
}
} // 最后是大于0的
DL_FOREACH_SAFE(_updatesPosList, entry, tmp)
{
if ((! entry->paused) && (! entry->markedForDeletion))
{
entry->callback(dt);
}
} // 这里循环的是自己定义定时器
for (tHashTimerEntry *elt = _hashForTimers; elt != nullptr; )
{
_currentTarget = elt;
_currentTargetSalvaged = false; if (! _currentTarget->paused)
{
// 遍历当前对象附属的全部定时器
for (elt->timerIndex = 0; elt->timerIndex < elt->timers->num; ++(elt->timerIndex))
{
elt->currentTimer = (Timer*)(elt->timers->arr[elt->timerIndex]);
elt->currentTimerSalvaged = false; //其实在这里运行真正的回调
elt->currentTimer->update(dt); if (elt->currentTimerSalvaged)
{
// 当定时器结束任务了。就应该释放掉
elt->currentTimer->release();
} elt->currentTimer = nullptr;
}
} // 指向链表的下一对象
elt = (tHashTimerEntry *)elt->hh.next; // 当对象的全部定时器已经运行完毕。而且对象附属的定时器为空,则将对象从哈希链表中移除
if (_currentTargetSalvaged && _currentTarget->timers->num == 0)
{
removeHashElement(_currentTarget);
}
} // 移除全部标记为删除的优先级小于0的update定时器元素
DL_FOREACH_SAFE(_updatesNegList, entry, tmp)
{
if (entry->markedForDeletion)
{
this->removeUpdateFromHash(entry);
}
} // 移除全部标记为删除的优先级等于0的update定时器元素
DL_FOREACH_SAFE(_updates0List, entry, tmp)
{
if (entry->markedForDeletion)
{
this->removeUpdateFromHash(entry);
}
} // 移除全部标记为删除的优先级大于0的update定时器元素
DL_FOREACH_SAFE(_updatesPosList, entry, tmp)
{
if (entry->markedForDeletion)
{
this->removeUpdateFromHash(entry);
}
} _updateHashLocked = false;
_currentTarget = nullptr;
}

有三个部分值得注意的:

  1. update定时器优先调用
  2. update定时器中优先级越低的越优先调用
  3. 自己定义定时器的回调在elt->currentTimer->update(dt);中运行
关于第三点,我们继续分析这个update函数

void Timer::update(float dt)
{
// 初次运行 会进入到这个if中初始化
if (_elapsed == -1)
{
_elapsed = 0; //已运行时间
_timesExecuted = 0; //初始化反复次数
}
else
{
if (_runForever && !_useDelay)
{//循环延时函数
_elapsed += dt;
if (_elapsed >= _interval)
{
trigger(); //真正的回调 _elapsed = 0;
}
}
else
{//advanced usage
_elapsed += dt;
if (_useDelay) //延时
{
if( _elapsed >= _delay )
{
trigger(); //真正的回调 _elapsed = _elapsed - _delay;
_timesExecuted += 1;
_useDelay = false;
}
}
else //每帧调用
{
if (_elapsed >= _interval)
{
trigger(); //真正的回调 _elapsed = 0;
_timesExecuted += 1; }
} //回调完毕,运行取消函数
if (!_runForever && _timesExecuted > _repeat)
{ //unschedule timer
cancel();
}
}
}
}

这一大段代码的逻辑很清晰。延时函数主要分为,永远循环的延时函数,有限循环的延迟函数。而有限循环的延迟函数里面依据优化的不同能够分为每帧调用的和固定时间调用的。上述代码就是依据这个分类进行优化的。

实际上核心的函数在

trigger();
cancel();

第一个函数是真正的回调运行的函数,第二个函数是去掉运行的函数

void TimerTargetSelector::trigger()
{
if (_target && _selector)
{
(_target->*_selector)(_elapsed);
}
} void TimerTargetSelector::cancel()
{
_scheduler->unschedule(_selector, _target);
}

以上就是定时器的实现原理分析的全过程,定时器的实如今文章中我感觉还是说的不是非常清楚。真正去代码中自己走一遍应该会更加明了,对以后的应用也会更得心应手。

【深入了解cocos2d-x 3.x】定时器(scheduler)的使用和原理探究(3)的更多相关文章

  1. Linux时间子系统之六:高精度定时器(HRTIMER)的原理和实现

    转自:http://blog.csdn.net/droidphone/article/details/8074892 上一篇文章,我介绍了传统的低分辨率定时器的实现原理.而随着内核的不断演进,大牛们已 ...

  2. cocos2d调度器(定时执行某函数)

    调度器(scheduler) 继承关系 原理介绍 Cocos2d-x调度器为游戏提供定时事件和定时调用服务.所有Node对象都知道如何调度和取消调度事件,使用调度器有几个好处: 每当Node不再可见或 ...

  3. 【转】 实现 Cocos2d-x 全局定时器

    转自:http://www.tairan.com/archives/3998 cocos2d-x 中有自己的定时器实现,一般用法是在场景,层等内部实现,定时器的生命周期随着它们的消亡而消亡,就运行周期 ...

  4. mysql的 深度使用 - 游标 , 定时器, 触发器 的使用 ?

    游标 叶叫做 光标; 只能使用在 mysql的 存储过程 或函数中! 游标的概念? 为什么要使用 游标? 什么叫 定时器, 就是事件 event! 是在 mysql 5.0以上的版本中, 才能使用支持 ...

  5. (5)调度器(scheduler)

    继承关系 原理介绍 Cocos2d-x调度器为游戏提供定时事件和定时调用服务.所有Node对象都知道如何调度和取消调度事件,使用调度器有几个好处: 每当Node不再可见或已从场景中移除时,调度器会停止 ...

  6. STM32之通用定时器

    广大的互联网的大家早上中午晚上..又好..没错了..我又来了..写博客不是定时的..为什么我要提写博客不是定时的呢??聪明的人又猜到我要说什么了吧.有前途.其实我还是第一次听到定时器有通用和高级之分的 ...

  7. JavaScript定时器的工作原理(翻译)

    JavaScript定时器的工作原理(翻译) 标签(空格分隔): JavaScript定时器 最近在看ajax原理的时候,看到了一篇国外的文章,讲解了JavaScript定时器的工作原理,帮助我很好的 ...

  8. javascript中的定时器

    本文地址:[http://www.xiabingbao.com/javascript/2015/04/20/javascript-timer/] 在以前的文章[javascript中的定时器]中,简单 ...

  9. Go定时器--Timer

    目录 前言 Timer 定时器 简介 使用场景 1. 设定超时时间 2. 延迟执行某个方法 Timer对外接口 1. 创建定时器 2. 停止定时器 3. 重置定时器 其他接口 1. After() 2 ...

随机推荐

  1. Excel.Application手册

    ----转载:http://blog.csdn.net/xxfigo/article/details/6618129 定制模块行为(1) Option Explicit '强制对模块内所有变量进行声明 ...

  2. break、continue和goto 三者作用介绍

    跳跃语句 由于break.continue和goto语句有助于跳跃到代码中的某个特定语句,因此它们属于跳跃语句.下面是这三个语句的介绍. ①break语句 这个语句常与switch语句联合使用:但是, ...

  3. birt 运行环境搭建(部署到tomcat)

    最近一直在研究eclipse的birt,各种坑~~~~(>_<)~~~~. Requirements:tomcat version:7.0,birt-runtime-4.6.0-20160 ...

  4. c++面试(二)

    1.宏参数的连接 #define CONS(a,b) (int)(a##e##b) CONS(2,3) =>2e3 =2000 2.const int b=10; int c=20; const ...

  5. Js 旋转平滑特效

    效果图 源码 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www. ...

  6. windows phone使用sharpcompress进行解压压缩文件

    在做移动端时,当我们需要从服务器获得多个文件时,为了节约流量,服务器一般会返回一个压缩包,那我们就是下载完成后,在手机中进行解压到指定位置 SharpCompress就是可以在手机中进行解压一个类库( ...

  7. 试用ubuntu-12.04.3-desktop-amd64(二)

    首先说明,采用主机+虚拟机+ubuntu的形式,更具体的则为Win7-64bit + VMWare + ubuntu-12.04.3-desktop-amd64 进入ubuntu后首先考虑到的就是怎么 ...

  8. Linux makefile 教程 非常详细,且易懂(转)

    转自:http://blog.chinaunix.net/uid-27717694-id-3696246.html 原文地址:Linux makefile 教程 非常详细,且易懂 作者:Deem_pa ...

  9. 50 Pow(x, n)(求x的n次方Medium)

    题目意思:x为double,n为int,求x的n次方 思路分析:直接求,注意临界条件 class Solution { public: double myPow(double x, int n) { ...

  10. log4net编译后命名空间找不到的问题

    log4net编译后命名空间找不到的问题问题:工程A添加dll引用后,可以在代码中引入log4net的命名空间.工程B引用A.A能编译成功,B不能编译成功,提示找不到命名空间. 解决: 点击项目属性- ...