二叉堆

二叉堆是优先队列很普遍的一种实现,它又分为最小堆最大堆,最小堆和最大堆都是完全二叉树。
其结构体定义如下:

struct HeapStruct {
    int Capacity;    //堆的最大容量
    int Size;         //堆的当前结点数
    ElementType *Elements;   //存放堆结点的数组
}

typedef struct HeapStruct *PriorityQueue;

二叉堆的结点可以保存在一个数组中。
1、如果从数组的索引1开始存放堆结点,那么对于数组中任意位置i上的元素,其左儿子在位置2i上,右儿子在左儿子后的单元(2i + 1)中
2、如果从数组的索引0开始存放堆结点,那么对于数组中任意位置i上的元素,其左儿子在位置(2i+1)上,右儿子在左儿子后的单元(2i+2)中。

在下面的代码中,统一采用第一种方式存放堆结点。你可以通过下面的代码注意到,用第一种方式给编程带来的好处。

最小堆:父结点的键值总是小于或等于任何一个子节点的键值。

(1)插入操作Insert,采用制造“空穴”,上滤的方式——“空穴”在堆中不断上移直到找出正确的位置。插入操作复杂度O(logN)。

void Insert(ElementType X, PriorityQueue H) {
    int i;

    if( IsFull(H) ) {      //判断堆是否已满
        Error( "Priority queue is full" );
        return ;
    }

    for( i = ++ H->Size; H->Elements[i/2] > X; i = i/2) {
        H->Elements[i] = H->Elements[i/2] ; 
    }
    H->Elements[i] = X ;
}

(2)删除最小元素(即根�结点)DeleteMin,下滤的方式。删除最小元素操作的复杂度O(logN),因为删除后涉及重建堆。

ElementType DeleteMin(PriorityQueue H) {
    int i, Child ;
    ElementType MinElement, LastElement;

    if( IsEmpty(H) ) {        //判断堆是否为空
        Error( "Priority queue is empty" );
        return H->Elements[0];
    }
    MinElement = H->Elements[1];
    LastElement = H->Elements[H->Size--]; //在�根结点删除了第一个堆结点,因此在根结点制造了一个“空穴”,
先保存最后一个堆结点,堆大小减1。下面的�代码是把LastElement放到合适的地方。

    for( i =1; i*2 <= H->Size; i= Child) {  //下滤过程
        
        //找较小的一个儿子结点
        Child = i * 2;            //左儿子结点
        if(Child != H->Size && H->Elements[Child + 1] < H->Elements[Child] )
            Child++;               //如果左儿子不是新堆的最后一个结点,且左儿子大于右儿子,我们选择右儿子
        
        //判断较小的儿子结点是否小于LastElement。若是,把该较小的儿子结点上移到自己的父结点;
        //若不是,退出循环。(即LastElement比儿子结点都小,可以作为它们的父结点插入)
        if(LastElement > H->Elements[Child] ) 
            H->Elements[i] = H->Elements[Child];
        else
            break;
    }
    H->Elements[i] = LastElement ; 
    return MinElement;    //返回被删除的根节点
}

(3)如果仅仅是�要获得最小值,那么可以在常数时间完成O(1)。

</br>

最大堆:父结点的键值总是�大于或等于任何一个子节点的键值。

(1)插入操作Insert,采用制造“空穴”,上滤的方式——“空穴”在堆中不断上移直到找出正确的位置。插入操作复杂度O(logN)。

void Insert(ElementType X, PriorityQueue H) {
    int i ;
    
    if( IsFull(H) ) {
        Error( "Priority queue is full" );
        return ;
    }

    for(i=++H->Size; H->Elements[i/2] < X; i = i/2) {
        H->Elements[i] = H->Elements[i/2];
    }
    H->Elements[i] = X;
}

对比最小堆的插入操作,可看到只是把for循环的条件">X"改为"<X"。意思是只要X比它的父结点大,就一直上滤。

(2)删除最大元素(即根�结点)DeleteMax,下滤的方式。删除最大元素操作的复杂度O(logN),因为删除后涉及重建堆。

ElementType DeleteMax(PriorityQueue H) {
    int i, Child;
    ElementType MaxElement, LastElement;

    if( IsEmpty(H) ) {
        Error( "Priority queue is empty" );
        return H->Elements[0];
    }

    MaxElement = H->Elements[1];
    LastElement = H->Elements[H->Size--];

    for(i=1; 2*i <=H-Size; i = Child) {
        Child = i*2;
*        if(Child != H->Size && H->Elements[Child+1] > H->Elements[Child])
            Child++;

*        if(LastElement < H->Elements[Child] )
            H->Elements[i] = H->Elements[Child];
        else
            break;
    } 
    H->Elements[i] = LastElement;
    return MaxElement;
}

与最小堆的删除最小元素操作相比,有2处条件判断发生改变,已用*号标出,读者可以自行体会。
(3)如果仅仅是要获得最大值,那么可以在常数时间完成O(1)。

</br>

二叉堆的应用——堆排序:

用最大堆完成堆排序,每次DeleteMax花费时间O(logN),对N个元素进行排序就是O(NlogN)。
另外建立二叉堆花费的时间为O(N)。不解可参考->《为什么堆排序构建堆的时间复杂度是N》
所以堆排序的时间复杂度为O(NlogN)+O(N) = O(NlogN),因为是就地排序,所以空间复杂度为O(1)。

void HeapAdjust(int *a, int i, int size) {   //调整堆
    int Child;
    int tmp;
    
    for(tmp = a[i]; 2*i+1 < size; i = Child) {
        Child = 2*i+1;   //左儿子
        if(Child != size-1 && a[Child+1] > a[Child])   //如果右儿子比左儿子大,取右儿子
            Child++;
    
        if(tmp < a[Child])
            a[i] = a[Child];
        else
          break;
    }
    a[i] = tmp;
}

void HeapSort(int *a, int size) {    //堆排序主例程
    int i;
    for(i = size/2; i>=0; i--) {    //构建堆
        HeapAdjust(a, i, size);
    }

    for(i = size-1; i>0; i--) {
        int temp = a[i];   //交换堆顶和最后一个元素,即每次将剩余元素中的最大者放到最后面 
        a[i] = a[0];
        a[0] = temp;
        HeapAdjust(a, 0, i);  //重新调整堆顶节点成为大顶堆
    }
    
}

注:堆排序代码的堆结点从数组索引0开始

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 211,423评论 6 491
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,147评论 2 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,019评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,443评论 1 283
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,535评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,798评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,941评论 3 407
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,704评论 0 266
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,152评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,494评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,629评论 1 340
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,295评论 4 329
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,901评论 3 313
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,742评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,978评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,333评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,499评论 2 348

推荐阅读更多精彩内容

  • 我们有意调整了排序的顺序,最后讲这个堆排序。不是因为它很难,而是它涉及到了基本的数据结构知识。 堆,又名“优先队列...
    吃个小烧饼阅读 389评论 0 3
  • 二叉堆 堆有序定义:当一颗二叉树的每个节点都大于等于它的两个子节点时, 被称为堆有序。二叉堆定义: 二叉堆是一组能...
    leoLy阅读 2,054评论 0 2
  • 一、优先队列 1.简单介绍 优先队列是一种抽象的数据结构,它与我们生活中的许多场景息息相关。比如我们的电脑或者手机...
    丶legend阅读 916评论 0 0
  • 1.什么是二叉堆 二叉堆是一种特殊的堆,二叉堆是完全二元树(二叉树)或者是近似完全二元树(二叉树)。二叉堆有两种:...
    RavenX阅读 2,171评论 0 3
  • 什么是堆? 如图所示:当父结点的值,都比其子结点的值小时,就是小根堆,一个小根堆中,最小的值肯定在堆顶。 同理,当...
    wayyyy阅读 542评论 0 0