上篇说到定时器的使用方法,这篇主要分析它的实现原理。

1.哈希链表

cocos2dx封装了一个结构体,叫做UT_hash_handle,只要在自定义的结构体中声明这个结构体变量,就实现了哈希链表,并且能使用一系列的哈希链表专用的宏。这个结构体的具体实现如下:
typedef struct UT_hash_handle {
   struct UT_hash_table *tbl;
   void *prev;                       /* prev element in app order      */
   void *next;                       /* next element in app order      */
   struct UT_hash_handle *hh_prev;   /* prevIoUs hh in bucket order    */
   struct UT_hash_handle *hh_next;   /* next hh in bucket order        */
   void *key;                        /* ptr to enclosing struct's key  */
   unsigned keylen;                  /* enclosing struct's key len     */
   unsigned hashv;                   /* result of hash-fcn(key)        */
} UT_hash_handle;

这个结构体主要实现的是一个双向链表,具体实现哈希验证的还要看UT_hash_table 结构体
typedef struct UT_hash_table {
   UT_hash_bucket *buckets;
   unsigned num_buckets,log2_num_buckets;
   unsigned num_items;
   struct UT_hash_handle *tail; /* tail hh in app order,for fast append    */
   ptrdiff_t hho; /* hash handle offset (byte pos of hash handle in element */

   /* in an ideal situation (all buckets used equally),no bucket would have
    * more than ceil(#items/#buckets) items. that's the ideal chain length. */
   unsigned ideal_chain_maxlen;

   /* nonideal_items is the number of items in the hash whose chain position
    * exceeds the ideal chain maxlen. these items pay the penalty for an uneven
    * hash distribution; reaching them in a chain traversal takes >ideal steps */
   unsigned nonideal_items;

   /* ineffective expands occur when a bucket doubling was performed,but 
    * afterward,more than half the items in the hash had nonideal chain
    * positions. If this happens on two consecutive expansions we inhibit any
    * further expansion,as it's not helping; this happens when the hash
    * function isn't a good fit for the key domain. When expansion is inhibited
    * the hash will still work,albeit no longer in constant time. */
   unsigned ineff_expands,noexpand;

   uint32_t signature; /* used only to find hash tables in external analysis */
#ifdef HASH_BLOOM
   uint32_t bloom_sig; /* used only to test bloom exists in external analysis */
   uint8_t *bloom_bv;
   char bloom_nbits;
#endif

} UT_hash_table;

然后看看与哈希链表相关的宏定义,使用这些宏能很方便的插入链表,删除链表,查找链表。
/**
 * 查找元素
 * head:哈希链表的头指针
 * findptr:要查找的元素指针
 * out:查找结果
 */
HASH_FIND_PTR(head,findptr,out) 
/**
 * 添加元素
 * head:哈希链表的头指针
 * ptrfield:要添加的元素指针
 * add:要添加的哈希链表元素
 */
HASH_ADD_PTR(head,ptrfield,add) 
/**
 * 替换元素
 * head:哈希链表的头指针
 * ptrfield:要替换的元素指针
 * add:要替换的哈希链表元素
 */
HASH_REPLACE_PTR(head,add)
/**
 * 删除
 * head:哈希链表的头指针
 * delptr:要删除的元素指针
 */
HASH_DEL(head,delptr) 

以上是引擎中实现的哈希链表的相关知识,接下来再看看与定时器相关的哈希链表。定时器的实现中,将一个定时器存储在哈希链表中,那么在scheduler是如何实现以后哈希链表的结构体的呢?如下:
// 不同优先级的update定时器的双向链表
typedef struct _listEntry
{
    struct _listEntry   *prev,*next;
    ccSchedulerFunc     callback;
    void                *target;
    int                 priority;
    bool                paused;
    bool                markedForDeletion; // selector will no longer be called and entry will be removed at end of the next tick
} tListEntry;
//内置的update定时器
typedef struct _hashUpdateEntry
{
    tListEntry          **list;        // Which list does it belong to ?
    tListEntry          *entry;        // entry in the list
    void                *target;
    ccSchedulerFunc     callback;
    UT_hash_handle      hh;
} tHashUpdateEntry;

// 自定义定时器
typedef struct _hashSelectorEntry
{
    ccArray             *timers;
    void                *target;
    int                 timerIndex;
    Timer               *currentTimer;
    bool                currentTimerSalvaged;
    bool                paused;
    UT_hash_handle      hh;
} tHashTimerEntry;

以上就是相关的哈希链表的知识,接下来从定义定时器的函数Node::schedule中一步一步的分析定时器是如何加入到哈希链表中的。

2.如何定义自定义定时器

首先,上一篇文章中说到了很多个自定义定时器的函数,但是最终会调用的函数只有两个,分别是
    /**
     * 定义一个自定义的定时器
	 * selector:回调函数
	 * interval:重复间隔时间,重复执行间隔的时间,如果传入0,则表示每帧调用
	 * repeat:重复运行次数,如果传入CC_REPEAT_FOREVER则表示无限循环
	 * delay:延时秒数,延迟delay秒开始执行第一次回调
     */
    void schedule(SEL_SCHEDULE selector,float interval,unsigned int repeat,float delay);
	
    /**
     * 使用lambda函数定义一个自定义定时器
     * callback:lambda函数
	 * interval:重复间隔时间,重复执行间隔的时间,如果传入0,则表示每帧调用
	 * repeat:重复运行次数,如果传入CC_REPEAT_FOREVER则表示无限循环
	 * delay:延时秒数,延迟delay秒开始执行第一次回调
     * key:lambda函数的Key,用于取消定时器
     * @lua NA
     */
    void schedule(const std::function<void(float)>& callback,float delay,const std::string &key);

本文从传统的定义定时器的方法入手,也就是第一个方法。接下来看看这个方法的实现:
void Node::schedule(SEL_SCHEDULE selector,float delay)
{
    CCASSERT( selector,"Argument must be non-nil");
    CCASSERT( interval >=0,"Argument must be positive");

    _scheduler->schedule(selector,this,interval,repeat,delay,!_running);
}

看到其实还是调用_scheduler的schedule方法,那么_scheduler又是个什么鬼?
Scheduler *_scheduler;          ///< scheduler used to schedule timers and updates
查看定义可以知道是一个Scheduler 的指针,但是这个指针从哪里来?在构造函数中有真相
Node::Node(void)
{
    // set default scheduler and actionManager
    _director = Director::getInstance();
    _scheduler = _director->getScheduler();
    _scheduler->retain();
}

是从导演类中引用的。这一块暂时我们不管,接下来深入到_scheduler->schedule函数中分析,如下是函数的具体实现
void Scheduler::schedule(SEL_SCHEDULE selector,Ref *target,bool paused)
{
    CCASSERT(target,"Argument target must be non-nullptr");
    
	//定义并且查找链表元素
    tHashTimerEntry *element = nullptr;
    HASH_FIND_PTR(_hashForTimers,&target,element);
    
	//没找到
    if (! element)
    {
		//创建一个链表元素
        element = (tHashTimerEntry *)calloc(sizeof(*element),1);
        element->target = target;
        
		//添加到哈希链表中
        HASH_ADD_PTR(_hashForTimers,target,element);
        
        // Is this the 1st element ? Then set the pause level to all the selectors of this target
        element->paused = paused;
    }
    else
    {
        CCASSERT(element->paused == paused,"");
    }
    
	//检查这个元素的定时器数组,如果数组为空 则new 10个数组出来备用
    if (element->timers == nullptr)
    {
        element->timers = ccArrayNew(10);
    }
    else
    {
		//循环查找定时器数组,看看是不是曾经定义过相同的定时器,如果定义过,则只需要修改定时器的间隔时间
        for (int i = 0; i < element->timers->num; ++i)
        {
            TimerTargetSelector *timer = dynamic_cast<TimerTargetSelector*>(element->timers->arr[i]);
            
            if (timer && selector == timer->getSelector())
            {
                cclOG("CCScheduler#scheduleSelector. Selector already scheduled. Updating interval from: %.4f to %.4f",timer->getInterval(),interval);
                timer->setInterval(interval);
                return;
            }
        }
		//扩展1个定时器数组
        ccArrayEnsureExtraCapacity(element->timers,1);
    }
    
	//创建一个定时器,并且将定时器加入到当前链表指针的定时器数组中
    TimerTargetSelector *timer = new (std::nothrow) TimerTargetSelector();
    timer->initWithSelector(this,selector,delay);
    ccArrayAppendobject(element->timers,timer);
    timer->release();
}


这一段代码具体分析了如何将自定义定时器加入到链表中,并且在链表中的存储结构是怎么样的,接下来看看内置的Update定时器。

3.如何定义Update定时器

Update定时器的开启方法有两个,分别是:
    /**
     * 开启自带的update方法,这个方法会每帧执行一次,默认优先级为0,并且在所有自定义方法执行之前执行
     */
    void scheduleUpdate(void);

    /**
     * 开启自带的update方法,这个方法会每帧执行一次,设定的优先级越小,越优先执行
     */
    void scheduleUpdateWithPriority(int priority);
第一个方法实际上是直接调用第二个方法,并且把优先级设置为0,我们直接看第二个方法就可以了。

void Node::scheduleUpdateWithPriority(int priority)
{
    _scheduler->scheduleUpdate(this,priority,!_running);
}
具体调用还是要进入到_scheduler->scheduleUpdate。
/** Schedules the 'update' selector for a given target with a given priority.
     The 'update' selector will be called every frame.
     The lower the priority,the earlier it is called.
     @since v3.0
     @lua NA
     */
    template <class T>
    void scheduleUpdate(T *target,int priority,bool paused)
    {
        this->schedulePerFrame([target](float dt){
            target->update(dt);
        },paused);
    }

可以看到这里主要还是调用了一个schedulePerFrame函数,并且传入了一个lambda函数。这个函数实际上调用的是target->update,接下来走进schedulePerFrame看看它的实现:
void Scheduler::schedulePerFrame(const ccSchedulerFunc& callback,void *target,bool paused)
{
	//定义并且查找链表元素
    tHashUpdateEntry *hashElement = nullptr;
    HASH_FIND_PTR(_hashForUpdates,hashElement);
	
	//如果找到,就直接改优先级
    if (hashElement)
    {
        // 检查优先级是否改变
        if ((*hashElement->list)->priority != priority)
        {
			//检查是否被锁定
            if (_updateHashLocked)
            {
                cclOG("warning: you CANNOT change update priority in scheduled function");
                hashElement->entry->markedForDeletion = false;
                hashElement->entry->paused = paused;
                return;
            }
            else
            {
            	// 在这里先停止到update,后面会加回来 
                unscheduleUpdate(target);
            }
        }
        else
        {
            hashElement->entry->markedForDeletion = false;
            hashElement->entry->paused = paused;
            return;
        }
    }

    // 优先级为0,加入到_updates0List链表中,并且加入到_hashForUpdates表中
    if (priority == 0)
    {
        appendIn(&_updates0List,callback,paused);
    }
	// 优先级小于0,加入到_updatesNegList链表中,并且加入到_hashForUpdates表中
    else if (priority < 0)
    {
        priorityIn(&_updatesNegList,paused);
    }
	// 优先级大于0,加入到_updatesPosList链表中,并且加入到_hashForUpdates表中
    else
    {
        // priority > 0
        priorityIn(&_updatesPosList,paused);
    }
}
在这里看上去逻辑还是很清晰的,有两个函数要重点分析一下,分别是
void Scheduler::appendIn(_listEntry **list,const ccSchedulerFunc& callback,bool paused)
void Scheduler::priorityIn(tListEntry **list,bool paused)

第一个用于添加默认优先级,第二个函数用于添加指定优先级的。首先看添加默认优先级的。
void Scheduler::appendIn(_listEntry **list,bool paused)
{
	//创建一个链表元素
    tListEntry *listElement = new tListEntry();

    listElement->callback = callback;
    listElement->target = target;
    listElement->paused = paused;
    listElement->priority = 0;
    listElement->markedForDeletion = false;
	
	//添加到双向链表中
    DL_APPEND(*list,listElement);

    //创建一个哈希链表元素
    tHashUpdateEntry *hashElement = (tHashUpdateEntry *)calloc(sizeof(*hashElement),1);
    hashElement->target = target;
    hashElement->list = list;
    hashElement->entry = listElement;
	//添加到哈希链表中
    HASH_ADD_PTR(_hashForUpdates,hashElement);
}

接下来看另一个函数
void Scheduler::priorityIn(tListEntry **list,bool paused)
{
	//同上一个函数
    tListEntry *listElement = new tListEntry();

    listElement->callback = callback;
    listElement->target = target;
    listElement->priority = priority;
    listElement->paused = paused;
    listElement->next = listElement->prev = nullptr;
    listElement->markedForDeletion = false;

    //如果链表为空
    if (! *list)
    {
        DL_APPEND(*list,listElement);
    }
    else
    {
        bool added = false;
		//保证链表有序
        for (tListEntry *element = *list; element; element = element->next)
        {
			// 如果优先级小于当前元素的优先级,就在这个元素前面插入
            if (priority < element->priority)
            {
                if (element == *list)
                {
                    DL_PREPEND(*list,listElement);
                }
                else
                {
                    listElement->next = element;
                    listElement->prev = element->prev;

                    element->prev->next = listElement;
                    element->prev = listElement;
                }

                added = true;
                break;
            }
        }

        //如果新加入的优先级最低,则加入到链表的最后
        if (! added)
        {
            DL_APPEND(*list,listElement);
        }
    }

    //同上一个函数
    tHashUpdateEntry *hashElement = (tHashUpdateEntry *)calloc(sizeof(*hashElement),1);
    hashElement->target = target;
    hashElement->list = list;
    hashElement->entry = listElement;
    HASH_ADD_PTR(_hashForUpdates,hashElement);
}
本文简单的分析了哈希链表以及定时器的存储和添加,下一篇文章将分析定时器是如何运转起来的。

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

  1. 深度解析swift中的String

    String是我们最常用到的语言元素,swift中的String初看起来相当简洁、易用,真正大量使用时,却有点摸不着头脑。直到看完了这篇文章,才算真正的明白了String的奥妙之处。每个Character所占用的内存空间不定,注定了String不能用普通的数组来存储内容,实际用的是双向链表。String.Index既然String是个双向链表,那么,访问其中的某个元素,或者substring,就要用指针了。NSRange和RangeNsstring中对于字符串区间,可以用NSRange来表示,而Strin

  2. swift 定时器的使用

    参考:http://www.cnblogs.com/sxlfybb/p/3792611.htmlNSTimer有2个构造函数注意到,注释为未继承,所以两个构造不能用NSTimer有2个静态实例在swift中,找不到NSInvocation类,日了狗了还好有另一种使用静态函数的形式去实例化:

  3. Swift 中数组和链表的性能

    尽管如此,我觉得链表的例子非常有意思,而且值得实现和把玩,它有可能会提升数组reduce方法的性能。同时我认为Swift的一些额外特性很有趣:比如它的枚举可以灵活的在对象和具体方法中自由选择,以及“默认安全”。这本书未来的版本可能就会用Swift作为实现语言。拷贝数组消耗的时间是线性的。使用链表还有其他的代价——统计链表节点的个数所需要的时间是统计数组元素个数时间的两倍,因为遍历链表时的间接寻址方式是需要消耗时间的。

  4. swift算法实践2

    字符串hash算法Time33在效率和随机性两方面上俱佳。对于一个Hash函数,评价其优劣的标准应为随机性,即对任意一组标本,进入Hash表每一个单元之概率的平均程度,因为这个概率越平均,数据在表中的分布就越平均,表的空间利用率就越高。Times33的算法很简单,就是不断的乘33,见下面算法原型。

  5. 《Swift NSDictionary 的详细使用和部分方法介绍 和 哈希表散列)的阐述和解释 》

    /*《SwiftNSDictionary的详细使用和部分方法介绍和哈希表(散列)的阐述和解释》*//*第一步:我们首先,必须了解一个概念性的东西那就是:哈希哈希的主要解释是:哈希算法将任意长度的二进制值映射为较短的固定长度的二进制值,这个小的二进制值称为哈希值。2》哈希列表是跟进式变化的。作为线性数据结构与表格和队列等相比,哈希表无疑是查找速度比较快的一种。在哈希方法中使用的转换函数hash被称作哈希函数。按照此中算法构造出来的表叫做哈希表。

  6. swift算法手记-10

    所有操作都以对数随机化的时间进行。每个更高层都充当下面列表的"快速跑道",这里在层i中的元素按某个固定的概率p出现在层i+1中。1------4---61---3-4---6------91-2-3-4-5-6-7-8-9-10结构实例要查找一个目标元素,起步于头元素和顶层列表,并沿着每个链表搜索,直到到达小于或的等于目标的最后一个元素。通过跟踪起自目标直到到达在更高列表中出现的元素的反向查找路径,在每个链表中预期的步数显而易见是1/p。通过选择不同p值,就可以在查找代价和存储代价之间作出权衡。

  7. Swift - 时间控制器NSTimer每隔一定时间执行某个函数

  8. 早期Swift中Cocos2D初始化代码的重构

    但是遗憾的是Swift2.2中还是不支持Type的class属性关键字,只能用static,我们期待Swift3的改进吧!

  9. Swift/OC计时器使用方法

    .invalidate();timer=nil;}

  10. swift3.0 Timer 定时器

    贡献者:赵大财博客:https://my.oschina.net/zhaodacaiGitHub:https://github.com/zhaodacai邮箱:zhaodacai@yeah.comQQ:327532817=============================

随机推荐

  1. 【cocos2d-x 3.x 学习笔记】对象内存管理

    Cocos2d-x的内存管理cocos2d-x中使用的是上面的引用计数来管理内存,但是又增加了一些自己的特色。cocos2d-x中通过Ref类来实现引用计数,所有需要实现内存自动回收的类都应该继承自Ref类。下面是Ref类的定义:在cocos2d-x中创建对象通常有两种方式:这两中方式的差异可以参见我另一篇博文“对象创建方式讨论”。在cocos2d-x中提倡使用第二种方式,为了避免误用第一种方式,一般将构造函数设为protected或private。参考资料:[1]cocos2d-x高级开发教程2.3节[

  2. 利用cocos2dx 3.2开发消灭星星六如何在cocos2dx中显示中文

    由于编码的不同,在cocos2dx中的Label控件中如果放入中文字,往往会出现乱码。为了方便使用,我把这个从文档中获取中文字的方法放在一个头文件里面Chinese.h这里的tex_vec是cocos2dx提供的一个保存文档内容的一个容器。这里给出ChineseWords,xml的格式再看看ChineseWord的实现Chinese.cpp就这样,以后在需要用到中文字的地方,就先include这个头文件然后调用ChineseWord函数,获取一串中文字符串。

  3. 利用cocos2dx 3.2开发消灭星星七关于星星的算法

    在前面,我们已经在GameLayer中利用随机数初始化了一个StarMatrix,如果还不知道怎么创建星星矩阵请回去看看而且我们也讲了整个游戏的触摸事件的派发了。

  4. cocos2dx3.x 新手打包APK注意事项!

    这个在编译的时候就可以发现了比较好弄这只是我遇到的,其他的以后遇到再补充吧。。。以前被这两个问题坑了好久

  5. 利用cocos2dx 3.2开发消灭星星八游戏的结束判断与数据控制

    如果你看完之前的,那么你基本已经拥有一个消灭星星游戏的雏形。开始把剩下的两两互不相连的星星消去。那么如何判断是GameOver还是进入下一关呢。。其实游戏数据贯穿整个游戏,包括星星消除的时候要加到获得分数上,消去剩下两两不相连的星星的时候的加分政策等,因此如果前面没有做这一块的,最好回去搞一搞。

  6. 利用cocos2dx 3.2开发消灭星星九为游戏添加一些特效

    needClear是一个flag,当游戏判断不能再继续后,这个flag变为true,开始消除剩下的星星clearSumTime是一个累加器ONE_CLEAR_TIME就是每颗星星消除的时间2.连击加分信息一般消除一次星星都会有连击信息和加多少分的信息。其实这些combo标签就是一张图片,也是通过控制其属性或者runAction来实现。源码ComboEffect.hComboEffect.cpp4.消除星星粒子效果消除星星时,为了实现星星爆裂散落的效果,使用了cocos2d提供的粒子特效引擎对于粒子特效不了

  7. 02 Cocos2D-x引擎win7环境搭建及创建项目

    官网有搭建的文章,直接转载记录。环境搭建:本文介绍如何搭建Cocos2d-x3.2版本的开发环境。项目创建:一、通过命令创建项目前面搭建好环境后,怎样创建自己的Cocos2d-x项目呢?先来看看Cocos2d-x3.2的目录吧这就是Cocos2d-x3.2的目录。输入cocosnew项目名–p包名–lcpp–d路径回车就创建成功了例如:成功后,找到这个项目打开proj.win32目录下的Hello.slnF5成功了。

  8. 利用cocos2dx 3.2开发消灭星星十为游戏添加音效项目源码分享

    一个游戏,声音也是非常的重要,其实cocos2dx里面的简单音效引擎的使用是非常简单的。我这里只不过是用一个类对所有的音效进行管理罢了。Audio.hAudio.cpp好了,本系列教程到此结束,第一次写教程如有不对请见谅或指教,谢谢大家。最后附上整个项目的源代码点击打开链接

  9. 03 Helloworld

    程序都有一个入口点,在C++就是main函数了,打开main.cpp,代码如下:123456789101112131415161718#include"main.h"#include"AppDelegate.h"#include"cocos2d.h"USING_NS_CC;intAPIENTRY_tWinMain{UNREFERENCED_ParaMETER;UNREFERENCED_ParaMETER;//createtheapplicationinstanceAppDelegateapp;return

  10. MenuItemImage*图标菜单创建注意事项

    学习cocos2dx,看的是cocos2d-x3.x手游开发实例详解,这本书错误一大把,本着探索求知勇于发现错误改正错误的精神,我跟着书上的例子一起调试,当学习到场景切换这个小节的时候,出了个错误,卡了我好几个小时。

返回
顶部