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

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. HTML 学习笔记

    1HTML 标题(Heading)是通过 <h1> - <h6> 等标签进行定义的. 并且只有这6种标题,标题中加多个空格,和一个空格没区别,标题文字前后加默认空格会被去除. ...

  2. 用XMPP实现完整Android聊天项目

    简介 这是一个完整的xmpp的Android的项目服务端使用openfire3.9.3客户端使用Android4.2.2 集成第三方:百度地图sdkasmack.jaruniversal-image- ...

  3. hibernate_validator_06

    认证组(校验组) 校验组能够让你在验证的时候选择应用哪些约束条件. 这样在某些情况下( 例如向导 ) 就可以对每一步进行校验的时候, 选取对应这步的那些约束条件进行验证了. 校验组是通过可变参数传递给 ...

  4. 【USACO 1.4.2】时钟

    [题目描述] 考虑将如此安排在一个 3 x 3 行列中的九个时钟: |-------| |-------| |-------| | | | | | | | |---O | |---O | | O | ...

  5. 关于超链接自动提示的demo

    <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01//EN" "http://www.w3.org/TR/html4/stri ...

  6. JS 操作Dom节点之样式

    为了提高用户体验,我们经常会动态修改Dom节点的样式,各种浏览器差异比较大,我们如何应对?不断尝试,不断总结~! 1. style.getComputedStyle.currentStyle 内嵌样式 ...

  7. IIS6中ASP.NET实现对静态文件的授权控制

    后台使用html+ashx+js开发 在VS2008调试并未发现问题 发布到IIS6才发现不需要验证也能访问html文件 解决这个问题配置IIS即可了 方法如下: IIS配置:网站->属性-&g ...

  8. JS之路——数组对象

    String字符串对象 concat() 返回一个由两个数组合并组成的新数组 join() 返回一个由数组中的所有元素连接在一起的String对象 pop() 删除数组中最后一个元素 并返回该值 pu ...

  9. Python三元表达式

    我们知道Python没有三元表达式,但是我们通过技巧达到三元表达式的效果. 摘自<Dive Into Python>: 在Python 中,and 和 or 执行布尔逻辑演算,如你所期待的 ...

  10. Hadoop 学习笔记(一) HDFS API

    http://www.cnblogs.com/liuling/p/2013-6-17-01.html 这个也不错http://www.teamwiki.cn/hadoop/thrift thrift编 ...