timersmanager 解析
最近在看crtmp源代码,看到timersmanager 模块时感觉很难理解,花了不少时间反复思考该模块
的逻辑,现在思考的结果记录下来,方便以后查阅。
构造函数中将处理时间方法传进来,将_lastTime赋值为当前时间,将当前slot Index设置为0,Slot指针
赋为空,slot数目赋为0。slot可以理解为槽。
TimersManager::TimersManager(ProcessTimerEvent processTimerEvent) {
_processTimerEvent = processTimerEvent;
_lastTime = time(NULL);
_currentSlotIndex = 0;
_pSlots = NULL;
_slotsCount = 0;
}
析构时释放掉slot,无需多解释。
TimersManager::~TimersManager() {
if (_pSlots != NULL)
delete[] _pSlots;
}
移除处理事件方法,将id为eventTimerId的事件处理方法从所有的slot中移除掉
void TimersManager::RemoveTimer(uint32_t eventTimerId) {
for (uint32_t i = ; i < _slotsCount; i++) {
if (MAP_HAS1(_pSlots[i].timers, eventTimerId)) {
_pSlots[i].timers.erase(eventTimerId);
}
}
}
void TimersManager::AddTimer(TimerEvent& timerEvent) {
UpdatePeriods(timerEvent.period);
uint32_t min = ;
uint32_t startIndex = ;
for (uint32_t i = ; i < _slotsCount; i++) {
if (min > _pSlots[i].timers.size()) {
startIndex = i;
min = _pSlots[i].timers.size();
}
}
while (!MAP_HAS1(_pSlots[startIndex % _slotsCount].timers, timerEvent.id)) {
_pSlots[startIndex % _slotsCount].timers[timerEvent.id] = timerEvent;
startIndex += timerEvent.period;
}
}
void TimersManager::TimeElapsed(uint64_t currentTime) {
int64_t delta = currentTime - _lastTime;
_lastTime = currentTime;
if (delta <= || _slotsCount == )
return;
for (int32_t i = ; i < delta; i++) {
//每间隔period个槽里都会存放TimeEvent,比如某个TimeEvent的period为100,delta为9,最多
TimeEvent对应的事件只执行一次。
FOR_MAP(_pSlots[_currentSlotIndex % _slotsCount].timers, uint32_t, TimerEvent, j) {
_processTimerEvent(MAP_VAL(j));
}
_currentSlotIndex++;
}
}
void TimersManager::UpdatePeriods(uint32_t period) {
//如果 TimeEvent周期已经存在直接返回
if (MAP_HAS1(_periodsMap, period))
return;
_periodsMap[period] = period;
//将TimeEvent 周期放在period vector中
ADD_VECTOR_END(_periodsVector, period);
//计算所有period的最小公倍数,比如之前period为4,新period为3,总Slot Count 为4 * 3 = 12
uint32_t newSlotsCount = LCM(_periodsVector, );
if (newSlotsCount == )
newSlotsCount = period;
if (newSlotsCount == _slotsCount)
return;
Slot *pNewSlots = new Slot[newSlotsCount];
if (_slotsCount > ) {
//将Slot Count增加到30,本质上试讲之前的slot复制5份
for (uint32_t i = ; i < newSlotsCount; i++) {
pNewSlots[i] = _pSlots[i % _slotsCount];
}
delete[] _pSlots;
}
_pSlots = pNewSlots;
_slotsCount = newSlotsCount;
}
GCD最大公约数,LCM 求最大公倍数
uint32_t TimersManager::GCD(uint32_t a, uint32_t b) {
while (b != ) {
uint32_t t = b;
b = a % b;
a = t;
}
return a;
}
uint32_t TimersManager::LCM(uint32_t a, uint32_t b) {
if (a == || b == )
return ;
uint32_t result = a * b / GCD(a, b);
FINEST("a: %u; b: %u; r: %u", a, b, result);
return result;
}
uint32_t TimersManager::GCD(vector<uint32_t> numbers, uint32_t startIndex) {
if (numbers.size() <= )
return ;
if (numbers.size() <= startIndex)
return ;
if (numbers.size() - startIndex > ) {
return GCD(numbers[startIndex], GCD(numbers, startIndex + ));
} else {
return GCD(numbers[startIndex], numbers[startIndex + ]);
}
}
uint32_t TimersManager::LCM(vector<uint32_t> numbers, uint32_t startIndex) {
if (numbers.size() <= )
return ;
if (numbers.size() <= startIndex)
return ;
if (numbers.size() - startIndex > ) {
return LCM(numbers[startIndex], LCM(numbers, startIndex + ));
} else {
return LCM(numbers[startIndex], numbers[startIndex + ]);
}
}
/*
*
* TimersManager tm(NULL);
TimerEvent t1 = {2, 1, NULL};
TimerEvent t2 = {3, 2, NULL};
TimerEvent t3 = {3, 3, NULL};
TimerEvent t4 = {4, 4, NULL};
TimerEvent t5 = {3, 5, NULL};
TimerEvent t6 = {2, 6, NULL};
TimerEvent t7 = {4, 7, NULL};
tm.AddTimer(t1);
tm.AddTimer(t2);
tm.AddTimer(t3);
tm.AddTimer(t4);
tm.AddTimer(t5);
tm.AddTimer(t6);
tm.AddTimer(t7);
*
* */
如下图所示:第一个TimeEvent的周期为4,所以slot数目为4,也就是图1中的前4个slot。添加第二个TimeEvent的周期为3,UpdatePeriods
方法将前4个slot复制3份得到12个slot,这样TimeEvent1被放到1,5,9个slot中。接下来AddTimer方法会将TimeEvent2添加到slot中:第一
个添加的slot是slot_2,因为这个slot的TimeEvent总数为0,接下来添加的solt分别是slot_5,slot_8,slot_11,如果继续添加下去是(11 + 3)%
3 = 2,但slot_2已经添加过,各槽添加TimeEvent2结束。作者将slot总槽书设计为所有的period最小公倍数,在这里发送作用了。

图1
timersmanager 解析的更多相关文章
- 【原】Android热更新开源项目Tinker源码解析系列之三:so热更新
本系列将从以下三个方面对Tinker进行源码解析: Android热更新开源项目Tinker源码解析系列之一:Dex热更新 Android热更新开源项目Tinker源码解析系列之二:资源文件热更新 A ...
- .NET Core中的认证管理解析
.NET Core中的认证管理解析 0x00 问题来源 在新建.NET Core的Web项目时选择“使用个人用户账户”就可以创建一个带有用户和权限管理的项目,已经准备好了用户注册.登录等很多页面,也可 ...
- Html Agility Pack 解析Html
Hello 好久不见 哈哈,今天给大家分享一个解析Html的类库 Html Agility Pack.这个适用于想获取某网页里面的部分内容.今天就拿我的Csdn的博客列表来举例. 打开页面 用Fir ...
- 【原】Android热更新开源项目Tinker源码解析系列之一:Dex热更新
[原]Android热更新开源项目Tinker源码解析系列之一:Dex热更新 Tinker是微信的第一个开源项目,主要用于安卓应用bug的热修复和功能的迭代. Tinker github地址:http ...
- 【原】Android热更新开源项目Tinker源码解析系列之二:资源文件热更新
上一篇文章介绍了Dex文件的热更新流程,本文将会分析Tinker中对资源文件的热更新流程. 同Dex,资源文件的热更新同样包括三个部分:资源补丁生成,资源补丁合成及资源补丁加载. 本系列将从以下三个方 ...
- 多线程爬坑之路-Thread和Runable源码解析之基本方法的运用实例
前面的文章:多线程爬坑之路-学习多线程需要来了解哪些东西?(concurrent并发包的数据结构和线程池,Locks锁,Atomic原子类) 多线程爬坑之路-Thread和Runable源码解析 前面 ...
- Asp.Net WebApi核心对象解析(下篇)
在接着写Asp.Net WebApi核心对象解析(下篇)之前,还是一如既往的扯扯淡,元旦刚过,整个人还是处于晕的状态,一大早就来处理系统BUG,简直是坑爹(好在没让我元旦赶过来该BUG),队友挖的坑, ...
- 【知识必备】内存泄漏全解析,从此拒绝ANR,让OOM远离你的身边,跟内存泄漏say byebye
一.写在前面 对于C++来说,内存泄漏就是new出来的对象没有delete,俗称野指针:而对于java来说,就是new出来的Object放在Heap上无法被GC回收:而这里就把我之前的一篇内存泄漏的总 ...
- SQL Server 数据加密功能解析
SQL Server 数据加密功能解析 转载自: 腾云阁 https://www.qcloud.com/community/article/194 数据加密是数据库被破解.物理介质被盗.备份被窃取的最 ...
随机推荐
- 【风马一族_mysql】MySQL免安装版环境配置图文教程
mysql存放在某一个磁盘中(笔者使用E盘) 配置系统变量 打开 电脑的属性 点击 高级系统设置 选择 高级 点击 环境变量 选择 系统变量 点击 变量Path,追加 值 E:\mysql-5.6.2 ...
- Windows和Linux下Mysql 重置root 密码
Windows系统下一:进入Windows窗口cmd命令行下,停止mysql服务: net stop mysql二:进入mysql的安装目录下,进去bin目录mysqld-nt.exe --skip- ...
- 仿淘宝颜色属性选择展示代码(jQuery)
模仿淘宝商品选择颜色和尺寸的效果,即选择商品颜色和尺寸的时候,把选择的颜色和尺寸放到一个页面容器里面,不足之处,还望指教. <!DOCTYPE HTML> <html lang=&q ...
- 将Tab栏居中的方法
原始tab: 居中后的tab(边缘效果是截图的问题): 改变方法如下: 找到Android SlidingTabLayout源代码,在Android SlidingTabLayout源代码中有一个方法 ...
- C# 乘法口诀表的实现方法
using System; using System.Collections.Generic; using System.Linq; using System.Text; namespace 乘法运算 ...
- 002-python基础-hello-world
python hello.py 时,明确的指出 hello.py 脚本由 python 解释器来执行. 如果想要类似于执行shell脚本一样执行python脚本,例: ./hello.py ,那么就需 ...
- Python学习教程(learning Python)--2.3 Python自定义函数传参函数设计
Python里自定义子函数时,可以在调用时携带一些参数到子函数里去处理.具体用法结构如下: def func(arguments): statement statement etc. 定义子函数一定要 ...
- hdu 1029
#include"stdio.h" int main(void) { int n,x,y,t,i; while(scanf("%d",&n)!=EOF) ...
- 深度理解依赖注入(Dependence Injection)
前面的话:提到依赖注入,大家都会想到老马那篇经典的文章.其实,本文就是相当于对那篇文章的解读.所以,如果您对原文已经有了非常深刻的理解,完全不需要再看此文:但是,如果您和笔者一样,以前曾经看过,似乎看 ...
- How to create QTP Shared Object Repository
How to create QTP Shared Object Repository To create a shared object repository by performing follow ...