Libevent源码分析(一):最小堆

2024-09-05 12:48
文章标签 分析 源码 最小 libevent

本文主要是介绍Libevent源码分析(一):最小堆,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Libevent中的timeout事件是使用最小堆来管理维护的.代码位于<minheap-internal.h>.

看函数命名和代码风格应该是一个C++程序员,函数名都挺好懂的,只是下面这个结构体变量命名比较坑....

typedef struct min_heap
{struct event** p;unsigned n, a;//n队列元素的多少,a代表队列空间的大小.
} min_heap_t;

注释是我加的,这命名,n啊a啊的,鬼知道啥意思....必须吐槽一下.

 

先来说说什么是最小堆:

1.堆是一个二叉树

2.最小堆:父节点的值总是小于等于子节点

如下图:

上图圆圈旁边的数字代表其在数组中的下标.堆一般是用数组来存储的,也就是说实际存储结构是连续的,只是逻辑上是一棵树的结构.这样做的好处是很容易找到堆顶的元素,对Libevent来说,很容易就可以找到距当前时间最近的timeout事件.

 现在想想看我们要插入一个元素,我们要怎么移动数组中元素的位置,使其逻辑上仍然是一个小堆?结合下图很容易看出来:

1.假设我们要插入的元素为6大于其父节点的值2.则把元素放在数组相应的index上,插入完成.

 

2.假设我们要插入的为2小于其父节点的值3.则交换该节点与其父节点的值.对于下图来说,交换完毕后插入就算完成了.那要是交换完后发现index=2的元素还小于其父节点index=0的呢?就又得在一次交换,如此循环,直到达到根节点或是其不大于父节点.

到了这里我们看看libevent里的实现代码就很清楚了

复制代码
int min_heap_push(min_heap_t* s, struct event* e)
{if (min_heap_reserve(s, s->n + 1))return -1;min_heap_shift_up_(s, s->n++, e);return 0;
}//分配队列大小.n代表队列元素个数多少.
int min_heap_reserve(min_heap_t* s, unsigned n)
{if (s->a < n)    //队列大小不足元素个数,重新分配空间.
    {struct event** p;unsigned a = s->a ? s->a * 2 : 8;  //初始分配8个指针大小空间,否则原空间大小翻倍.if (a < n)a = n;   //翻倍后空间依旧不足,则分配n.if (!(p = (struct event**)mm_realloc(s->p, a * sizeof *p))) //重新分配内存return -1;s->p = p; //重新赋值队列地址及大小.s->a = a; //
    }return 0;
}void min_heap_shift_up_(min_heap_t* s, unsigned hole_index, struct event* e)
{unsigned parent = (hole_index - 1) / 2;while (hole_index && min_heap_elem_greater(s->p[parent], e)) //比父节点小或是到达根节点.则交换位置.循环.
    {(s->p[hole_index] = s->p[parent])->ev_timeout_pos.min_heap_idx = hole_index;hole_index = parent;parent = (hole_index - 1) / 2;}(s->p[hole_index] = e)->ev_timeout_pos.min_heap_idx = hole_index;
}
复制代码

 实际上作者写了一个比较通用的函数min_heap_shift_up(),与之相对应的还有min_heap_shift_down()

复制代码
void min_heap_shift_down_(min_heap_t* s, unsigned hole_index, struct event* e)
{unsigned min_child = 2 * (hole_index + 1);while (min_child <= s->n){//找出较小子节点min_child -= min_child == s->n || min_heap_elem_greater(s->p[min_child], s->p[min_child - 1]);//比子节点小正常.不需要再交换位置,跳出循环.if (!(min_heap_elem_greater(e, s->p[min_child])))break;//比子节点大,要交换位置(s->p[hole_index] = s->p[min_child])->ev_timeout_pos.min_heap_idx = hole_index;hole_index = min_child;min_child = 2 * (hole_index + 1);}(s->p[hole_index] = e)->ev_timeout_pos.min_heap_idx = hole_index;
}
复制代码

这里的hole_index是我们要填入某个值的下标,e是要填入的值.还是画图比较好理解:

当这个值(下标为hole_Index=1)比其父节点1(index = 0)小时,要向上移动调整.

当这个值(下标为hole_Index=1)比其最小子节点6(index = 3)还大时,要向下移动调整.

Libevent里是这么使用他们的:

复制代码
int min_heap_erase(min_heap_t* s, struct event* e)
{if (-1 != e->ev_timeout_pos.min_heap_idx){struct event *last = s->p[--s->n];//把最后一个值作为要填入hole_index的值unsigned parent = (e->ev_timeout_pos.min_heap_idx - 1) / 2;/* we replace e with the last element in the heap.  We might need toshift it upward if it is less than its parent, or downward if it isgreater than one or both its children. Since the children are knownto be less than the parent, it can't need to shift both up anddown. */if (e->ev_timeout_pos.min_heap_idx > 0 && min_heap_elem_greater(s->p[parent], last))min_heap_shift_up_(s, e->ev_timeout_pos.min_heap_idx, last);elsemin_heap_shift_down_(s, e->ev_timeout_pos.min_heap_idx, last);e->ev_timeout_pos.min_heap_idx = -1;return 0;}return -1;
}
复制代码
复制代码
struct event* min_heap_pop(min_heap_t* s)
{if (s->n){struct event* e = *s->p;min_heap_shift_down_(s, 0u, s->p[--s->n]);e->ev_timeout_pos.min_heap_idx = -1;return e;}return 0;
}
复制代码

 

最后总结一下,由于堆这种结构在逻辑上的这种二叉树的关系,其插入也好,删除也好,就是一个与父节点或是子节点比较然后调整位置,这一过程循环往复直到达到边界条件的过程.记住这一点,就不难写出代码了.

二叉树节点i:父节点为(i-1)/2.子节点为2i+1,2(i+1)。

这篇关于Libevent源码分析(一):最小堆的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1139012

相关文章

C++ Sort函数使用场景分析

《C++Sort函数使用场景分析》sort函数是algorithm库下的一个函数,sort函数是不稳定的,即大小相同的元素在排序后相对顺序可能发生改变,如果某些场景需要保持相同元素间的相对顺序,可使... 目录C++ Sort函数详解一、sort函数调用的两种方式二、sort函数使用场景三、sort函数排序

Java调用C++动态库超详细步骤讲解(附源码)

《Java调用C++动态库超详细步骤讲解(附源码)》C语言因其高效和接近硬件的特性,时常会被用在性能要求较高或者需要直接操作硬件的场合,:本文主要介绍Java调用C++动态库的相关资料,文中通过代... 目录一、直接调用C++库第一步:动态库生成(vs2017+qt5.12.10)第二步:Java调用C++

kotlin中const 和val的区别及使用场景分析

《kotlin中const和val的区别及使用场景分析》在Kotlin中,const和val都是用来声明常量的,但它们的使用场景和功能有所不同,下面给大家介绍kotlin中const和val的区别,... 目录kotlin中const 和val的区别1. val:2. const:二 代码示例1 Java

Go标准库常见错误分析和解决办法

《Go标准库常见错误分析和解决办法》Go语言的标准库为开发者提供了丰富且高效的工具,涵盖了从网络编程到文件操作等各个方面,然而,标准库虽好,使用不当却可能适得其反,正所谓工欲善其事,必先利其器,本文将... 目录1. 使用了错误的time.Duration2. time.After导致的内存泄漏3. jsO

Python实现无痛修改第三方库源码的方法详解

《Python实现无痛修改第三方库源码的方法详解》很多时候,我们下载的第三方库是不会有需求不满足的情况,但也有极少的情况,第三方库没有兼顾到需求,本文将介绍几个修改源码的操作,大家可以根据需求进行选择... 目录需求不符合模拟示例 1. 修改源文件2. 继承修改3. 猴子补丁4. 追踪局部变量需求不符合很

Spring事务中@Transactional注解不生效的原因分析与解决

《Spring事务中@Transactional注解不生效的原因分析与解决》在Spring框架中,@Transactional注解是管理数据库事务的核心方式,本文将深入分析事务自调用的底层原理,解释为... 目录1. 引言2. 事务自调用问题重现2.1 示例代码2.2 问题现象3. 为什么事务自调用会失效3

找不到Anaconda prompt终端的原因分析及解决方案

《找不到Anacondaprompt终端的原因分析及解决方案》因为anaconda还没有初始化,在安装anaconda的过程中,有一行是否要添加anaconda到菜单目录中,由于没有勾选,导致没有菜... 目录问题原因问http://www.chinasem.cn题解决安装了 Anaconda 却找不到 An

Spring定时任务只执行一次的原因分析与解决方案

《Spring定时任务只执行一次的原因分析与解决方案》在使用Spring的@Scheduled定时任务时,你是否遇到过任务只执行一次,后续不再触发的情况?这种情况可能由多种原因导致,如未启用调度、线程... 目录1. 问题背景2. Spring定时任务的基本用法3. 为什么定时任务只执行一次?3.1 未启用

C++ 各种map特点对比分析

《C++各种map特点对比分析》文章比较了C++中不同类型的map(如std::map,std::unordered_map,std::multimap,std::unordered_multima... 目录特点比较C++ 示例代码 ​​​​​​代码解释特点比较1. std::map底层实现:基于红黑

Spring、Spring Boot、Spring Cloud 的区别与联系分析

《Spring、SpringBoot、SpringCloud的区别与联系分析》Spring、SpringBoot和SpringCloud是Java开发中常用的框架,分别针对企业级应用开发、快速开... 目录1. Spring 框架2. Spring Boot3. Spring Cloud总结1. Sprin