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 数据加密是数据库被破解.物理介质被盗.备份被窃取的最 ...
随机推荐
- CentOS 5.5 下安装Countly Web Server过程记录
CentOS 5.5 下安装Countly Web Server过程记录 1. 系统更新与中文语言包安装 2. 基本环境配置: 2.1. NodeJS安装 依赖项安装 yum -y install g ...
- Mediator
#include <iostream> using namespace std; class ObjectA { public: void Whoami() { cout<<& ...
- phpcms v9 源码解析-1 index.php
这个是phpcms V9 的入口文件index.php. V9程序的执行绝大多数是从这个文件开始的,但不绝对,在项目下面的api.php和plugin.php是另外的入口文件,这里我们先不做深究. 在 ...
- ubuntu下svn使用指南
ubuntu下安装subversion客户端: sudo apt-get install subversion subversion-tools 详细请看 http://www.subversion. ...
- 【Weblogic】--Weblogic的部署方式和缓存
参考网址: http://dead-knight.iteye.com/blog/1938882 Weblogic11g部署web应用,有三种方式,非常简单,但是很多新手部署总是出现若干错误,不知道如何 ...
- UltraEdit中使用正则表达式
正则表达式 (UltraEdit Syntax): % 匹配行首 - 表明要搜索的字符串一定在行首. $ 匹配行尾 - 表明要搜索的字符串一定在行尾 ? 匹配除换行符外的任一单个字符. * 匹配任意个 ...
- Python之MySql操作
1.安装驱动 输入命令:pip install MySQL-python 2.直接使用驱动 #coding=utf-8 import MySQLdb conn= MySQLdb.connect( ho ...
- MongoDB工具简要说明
[mongodb@hadoop1 bin]$ pwd /usr/local/mongodb/bin [mongodb@hadoop1 bin]$ ls -l total 207696 -rwxr-xr ...
- 第三节:卸载AppDomain
AppDomain很出色的一个能力就是它允许卸载.卸载AppDomain会导致CLR卸载AppDomain中的所有程序集.还会释放AppDomain的Loader堆.为了卸载一个AppDomain,可 ...
- salt-ssh安装及简单使用
需要 salt-master 0.17以上版本支持 1.安装 相关依赖包可查看requirements.txt Jinja2 M2Crypto msgpack-python pycrypto PyYA ...