前言
- 文章属于旧有文章搬运, 之前在csdn上面
- 2019.11.6修改
Timer是定时器, 用途:
- 延时执行事件
- 周期性执行事件
参考官方文档Using Timers
Timer可以...
Timer能...
Timer...
...
编不下去了, 自己百度谷歌吧(手动狗头)
TimerManager
Timer的管理类是FTimerManager, 是一个全局的Timer管理类.
通常通过UWorld::GetTimerManager()获得
可以看到实际实现是先判断GameInstance是否存在, 存在返回UGameInstance的TimerManger, 否则返回UWorld自己的TimerManger
接着, 如上图, TimerManger在UWorld和UGameInstance的构造函数中创建, 调用构造函数, 不试图赋值UGameInstance
在TimerManger的构造函数中, 主要还是给InternalTime设0, 该值是TimerManger内部的一个独立时钟, 在TimerManger的每次Tick调用时计时.
在UWorld::Tick中调用FTimerManager的Tick函数实际执行Timer里面的操作
TimerManger的销毁是在UWorld::FinishDestroy和UGameInstance::FinishDestroy中
delete然后置nullptr, 简单粗暴.
Timer的执行
从堆栈中也可以FTimerManger::Tick被UWorld::Tick调用
如图, 这个函数体主要做了如下内容 :
- 先调用FTimerManager::HasBeenTickedThisFrame, 并维护LastTickedFrame避免一帧内被多次执行.
- 然后InternalTime计时, 增加DeltaTime偏移量
- 接着对ActiveTimerHeap和PendingTimerSet进行一定逻辑处理
ActiveTimerHeap是一个当前活跃的, 待处理的FTimerHandle数组
看while循环体主体逻辑:
不断取堆顶的FTimerHandle, 并得到对应的FTimerData.
如果该FTimerData的状态是可被移除, 那么移除, 继续
不断执行, 直到InternalTimer(内部的独立时钟)大于FTimerData的ExpireTime, 即定时器到时间了, 那么执行一定逻辑, 否则, 没有过时间, 就直接break跳出循环
因为ActiveTimerHeap是按照过期时间排序的
接着看, 如果定时器到时间了, 会将ActiveTimerHeap的堆顶元素移动置CurrentlyExecutingTimer(当前正在执行的Timer), 并更新FTimerData的Status为ETimerStatus::Executing
注意, 这里的CurrentlyExecutingTimer和上文中的TopHandler是同一个值, 所以改Top(FTImerData)修改的是同一Timer的信息
然后, 重点来了
划重点了!
划重点了!
划重点了!
上图圈起来的代码, CallCount(执行次数), 为了避免在两帧之间时间过长而导致丢掉一些定时器的间隔执行, 计算, 两帧之间实际需要执行多少次.
设一个Timer, 每0.1秒执行一次, Tick总共过了5秒, 有
- 定时器必定执行50次, 执行次数恒定
- 如果比较卡顿, 每0.5秒执行1帧, 即定时器可能也可以在同一帧执行多次
- 定时器是假定时, 即期望0.1s, 0.2s, 0.3s, 0.4s, 0.5s执行, 实际可能是0.5s, 0.5s, 0.5s, 0.5s, 0.5s执行5次
而Timer的执行, 实际上也就是绑定的代理执行. 此处略
// 可以参考代理相关的文章自己啃, 难度不大
在执行后有个更新指针的操作, 为了防止执行后无效, 这是为什么呢?
考验你C++功底的时候到了, 自行理解, 此处忽略(手动狗头)
在Timer执行完毕之后, 做清理.
如果Timer不循环, 执行一次, 那么完成使命了, 清理掉吧!
如果Timer代理无效了, 那么无法完成使命, 也清理掉吧!
否则, 嗯, 设置下次时间, 打回去, 接着干活(手动狗头)
// 所以循环Timer啊, Timer对应的代理实际对象啊, 多数是不需要人工维护移除的
// 即多数Timer绑定好, 不清除, 也没啥影响, 他会自动移除无效的Timer
最后再看PendingTimerSet的处理, 这是在当前Tick帧添加的定时器集合, 将里面的定时器添加到ActiveTimerHeap里面
即很明显, Timer最少也会延迟一帧调用
Timer的使用
参照上面若干函数, 最后都通过FTimerManger::InternalSetTimer函数SetTimerForNextTick相关基本同, 略
FTimerManger::InternalSetTimer中在设置Timer时, 会查找一下TimerHandle是否有效, 有效会清除.
接着就是创建一个FTimerData信息, 并维护, 具体略了
最后贴一张使用图, 一些基础知识就懒得提了.
// 基础知识 : TimerHandler清除, 获得剩余时间, 暂停和取消暂停, 是否存在等函数
// 看看头文件, 配合使用即可.
以及提一下这两个Timer的代理:
- DECLARE_DELEGATE(FTimerDelegate)
- DECLARE_DYNAMIC_DELEGATE(FTimerDynamicDelegate)
自行参考代理相关文章, 配合使用, 能玩出花(笑)
结语
- 骗赞, 骗评论了.